MySQL数据库主从复制和读写分离

news2025/1/10 16:11:29

目录

一、MySQL主从复制和读写分离理论

(一)读写分离

1.什么是读写分离

2.为什么要读写分离

3.什么时候要读写分离

4.读写分离原理

5.常见MySQL 读写分离

(1)基于程序代码内部实现

(2)基于中间代理层实现

① MySQL-Proxy

② Atlas

③ Amoeba

④ Mycat

(二)主从复制

1.mysql支持的复制类型

(1)STATEMENT:基于语句的复制

(2)ROW:基于行的复制

(3)MIXED:混合类型的复制

2.主从复制的工作过程

3.MySQL主从复制的同步模式

(1)异步复制

(2)全同步复制

(3)半同步复制

二、MySQL主从复制和读写分离实操

(一)搭建MySQL主从复制

1.主从服务器时间同步

(1)主服务器配置

(2)从服务器配置

​编辑

2.主服务器的mysql配置

3.从服务器的mysql配置

(1)修改配置文件

(2)登录数据库配置

4.验证主从复制

(1)在主服务器上创建库、表

(2)在从服务器中查看

(二)搭建半同步复制

1.主服务器配置

2.从服务器配置

3.查看半同步是否在运行

(1)主服务器上查看

(2)从服务器上查看

(3)在主库查询半同步状态

(三)搭建 MySQL读写分离

1.Amoeba服务器配置

(1)安装Java环境

(2)安装Amoeba软件

(3)配置 Amoeba读写分离,两个Slave读负载均衡

(4)配置amoeba服务

2.测试读写分离

(1)主服务器上创建库和表,并添加数据

(2)在两台从服务器上查看表内容

(3)在从服务器上添加数据

(4)在主服务器上查看


一、MySQL主从复制和读写分离理论

(一)读写分离

1.什么是读写分离

        读写分离,基本的原理是让主数据库处理事务性增、改、删操作(INSERT、UPDATE、DELETE),而从数据库处理SELECT查询操作。数据库复制被用来把事务性操作导致的变更同步到集群中的从数据库。

2.为什么要读写分离

         因为数据库的“写”(写10000条数据可能要3分钟)操作是比较耗时的,但是数据库的“读”(读10000条数据可能只要5秒钟),所以读写分离,解决的是,数据库的写入,影响了查询的效率。

3.什么时候要读写分离

        数据库不一定要读写分离,如果程序使用数据库较多时,而更新少,查询多的情况下会考虑使用。利用数据库主从同步,再通过读写分离可以分担数据库压力,提高性能。

4.读写分离原理

       读写分离就是只在主服务器上写,只在从服务器上读。基本的原理是让主数据库处理事务性操作,而从数据库处理 select 查询。数据库复制被用来把主数据库上事务性操作导致的变更同步到集群中的从数据库。

5.常见MySQL 读写分离

(1)基于程序代码内部实现

        在代码中根据 select、insert 进行路由分类,这类方法也是目前生产环境应用最广泛的。优点是性能较好,因为在程序代码中实现,不需要增加额外的设备为硬件开支;缺点是需要开发人员来实现,运维人员无从下手。但是并不是所有的应用都适合在程序代码中实现读写分离,像一些大型复杂的Java应用,如果在程序代码中实现读写分离对代码改动就较大。

(2)基于中间代理层实现

       代理一般位于客户端和服务器之间,代理服务器接到客户端请求后通过判断后转发到后端数据库,有以下代表性程序:

① MySQL-Proxy

       MySQL-Proxy 为 MySQL 开源项目,通过其自带的 lua 脚本进行SQL 判断。

② Atlas

       是由奇虎360的Web平台部基础架构团队开发维护的一个基于MySQL协议的数据中间层项目。它是在mysql-proxy 0.8.2版本的基础上,对其进行了优化,增加了一些新的功能特性。360内部使用Atlas运行的mysql业务,每天承载的读写请求数达几十亿条。支持事物以及存储过程。

③ Amoeba

       由陈思儒开发,作者曾就职于阿里巴巴。该程序由Java语言进行开发,阿里巴巴将其用于生产环境。但是它不支持事务和存储过程。

④ Mycat

       是一款流行的基于Java语言编写的数据库中间件,是一个实现了MySql协议的服务器,其核心功能是分库分表。配合数据库的主从模式还可以实现读写分离。

        由于使用MySQL Proxy 需要写大量的Lua脚本,这些Lua并不是现成的,而是需要自己去写。这对于并不熟悉MySQL Proxy 内置变量和MySQL Protocol 的人来说是非常困难的。Amoeba是一个非常容易使用、可移植性非常强的软件。因此它在生产环境中被广泛应用于数据库的代理层。

(二)主从复制

1.mysql支持的复制类型

(1)STATEMENT:基于语句的复制

       在服务器上执行sql语句,在从服务器上执行同样的语句,mysql默认采用基于语句的复制,执行效率高。

(2)ROW:基于行的复制

        把改变的内容复制过去,而不是把命令在从服务器上执行一遍。

(3)MIXED:混合类型的复制

        默认采用基于语句的复制,一旦发现基于语句无法精确复制时,就会采用基于行的复制。

2.主从复制的工作过程

(1)Master节点将数据的改变记录成二进制日志(bin log),当Master上的数据发生改变时,则将其改变写入二进制日志中。

(2)Slave节点会在一定时间间隔内对Master的二进制日志进行探测其是否发生改变,如果发生改变,则开始一个I/O线程请求 Master的二进制事件。

(3)同时Master节点为每个I/O线程启动一个dump线程,用于向其发送二进制事件,并保存至Slave节点本地的中继日志(Relay log)中,Slave节点将启动SQL线程从中继日志中读取二进制日志,在本地重放,即解析成 sql 语句逐一执行,使得其数据和 Master节点的保持一致,最后I/O线程和SQL线程将进入睡眠状态,等待下一次被唤醒。

3.MySQL主从复制的同步模式

(1)异步复制

       Asynchronous replication: MySQL默认的复制即是异步的,主库在执行完客户端提交的事务后会立即将结果返给客户端,并不关心从库是否已经接收并处理,这样就会有一个问题,主如果crash掉了,此时主上已经提交的事务可能并没有传到从上,如果此时,强行将从提升为主,可能导致新主上的数据不完整。

(2)全同步复制

       Fully synchronous replication:指当主库执行完一个事务,所有的从库都执行了该事务才返回给客户端。因为需要等待所有从库执行完该事务才能返回,所以全同步复制的性能必然会收到严重的影响。

(3)半同步复制

       Semisynchronous replication:介于异步复制和全同步复制之间,主库在执行完客户端提交的事务后不是立刻返回给客户端,而是等待至少一个从库接收到并写到relay log中才返回给客户端。相对于异步复制,半同步复制提高了数据的安全性,同时它也造成了一定程度的延迟,这个延迟最少是一个TCP/IP往返的时间。所以,半同步复制最好在低延时的网络中使用。

二、MySQL主从复制和读写分离实操

(一)搭建MySQL主从复制

1.主从服务器时间同步

(1)主服务器配置
yum install ntp -y
 
vim /etc/ntp.conf
  server 172.16.72.0	
  #设置本地是时钟源,注意修改网段
  fudge 172.16.72.0 stratum 8
  #设置时间层级为8(限制在15内)
 
service ntpd start

(2)从服务器配置

两台从服务器同样配置

yum install ntp ntpdate -y
systemctl start ntpd
/usr/sbin/ntpdate 172.16.72.40				
#进行时间同步
 
crontab -e
#添加周期性计划任务
*/30 * * * * /usr/sbin/ntpdate 172.16.72.40
#每隔30分钟与主服务器同步一次时间

2.主服务器的mysql配置

vim /etc/my.cnf
  server-id=1
  log-bin=mysql-bin		
  binlog_format=mixed
  #添加,主服务器开启二进制日志

  #选配项
  expire_logs_days=7	
  #设置二进制日志文件过期时间,默认值为0,表示logs不过期
  max_binlog_size=500M
  #设置二进制日志限制大小,如果超出给定值,日志就会发生滚动,默认值是1GB
  skip_slave_start=1
  #阻止从库崩溃后自动启动复制,崩溃后再自动复制可能会导致数据不一致的
 
  #"双1设置",数据写入最安全
  innodb_flush_log_at_trx_commit=1		
  #redo log(事务日志)的刷盘策略,每次事务提交时MySQL都会把事务日志缓存区的数据写入日志文件中,并 
  且刷新到磁盘中,该模式为系统默认
  sync_binlog=1							
  #在进行每1次事务提交(写入二进制日志)以后,Mysql将执行一次fsync的磁盘同步指令,将缓冲区数据刷新 
  到磁盘
 
systemctl restart mysqld
mysql -u root
grant replication slave on *.* to 'myslave'@'172.16.72.%' identified by '123'; 
#给从服务器授权
flush privileges;
show master status;

3.从服务器的mysql配置

两台从服务器一样配置

(1)修改配置文件
vim /etc/my.cnf
server-id = 2                       #修改,注意id与Master的不同,两个Slave的id也要不同
relay-log=relay-log-bin			    #开启中继日志,从主服务器上同步日志文件记录到本地
relay-log-index=relay-log-bin.index	#定义中继日志文件的位置和名称,一般和relay-log在同一目录
 
选配项
innodb_buffer_pool_size=2048M		
#用于缓存数据和索引的内存大小,让更多数据读写内存中完成,减少磁盘操作,可设置为服务器总可用内存的 70-80%
sync_binlog=0						
#MySQL不做任何强制性的磁盘刷新指令,而是依赖操作系统来刷新数据到磁盘
innodb_flush_log_at_trx_commit=2	
#每次事务log buffer会写入log file,但一秒一次刷新到磁盘
log-slave-updates=0					
#slave 从 master 复制的数据会写入二进制日志文件里,从库做为其他从库的主库时设置为 1
relay_log_recovery=1				
#当 slave 从库宕机后,假如 relay-log 损坏了,导致一部分中继日志没有处理,则自动放弃所有未执行的 relay-log, 并且重新从 master 上获取日志,这样就保证了 relay-log 的完整性。默认情况下该功能是关闭的,将 relay_log_recovery 的值设置为 1 时, 可在 slave 从库上开启该功能,建议开启。
 
systemctl restart mysqld

(2)登录数据库配置
mysql -uroot -p123 
 
change master to master_host='172.16.72.40',master_user='myslave',master_password='123',master_log_file='mysql-bin.000001',master_log_pos=3123;
start slave;
show slave status\G
 
//确保 IO 和 SQL 线程都是 Yes,代表同步正常。
Slave_IO_Running: Yes				#负责与主机的io通信
Slave_SQL_Running: Yes				#负责自己的slave mysql进程
 
#一般 Slave_IO_Running: No 的可能性:
1、网络不通
2、my.cnf配置有问题
3、密码、file文件名、pos偏移量不对
4、防火墙没有关闭

4.验证主从复制

(1)在主服务器上创建库、表

(2)在从服务器中查看

(二)搭建半同步复制

1.主服务器配置

vim /etc/my.cnf	
#在 [mysqld] 区域添加下面内容
 
plugin-load=rpl_semi_sync_master=semisync_master.so	
#加载mysql半同步复制的插件
rpl_semi_sync_master_enabled=ON	
#或者设置为"1",即开启半同步复制功能
rpl-semi-sync-master-timeout=1000
#超时时间为1000ms,即1s
 
systemctl restart mysqld

2.从服务器配置

vim /etc/my.cnf
 
plugin-load=rpl_semi_sync_slave=semisync_slave.so
rpl_semi_sync_slave_enabled=ON
 
systemctl restart mysqld

3.查看半同步是否在运行

(1)主服务器上查看
show status like 'Rpl_semi_sync_master_status';
show variables like 'rpl_semi_sync_master_timeout';

(2)从服务器上查看
show status like 'Rpl_semi_sync_slave_status';
 
stop slave io_thread;
#关闭数据库上的IO线程
start salve io_thread;
#启动数据库上的IO线程

(3)在主库查询半同步状态
show status like '%Rpl_semi%';
 
Rpl_semi_sync_master_status
#表示当前是异步模式还是半同步模式,on为半同步

(三)搭建 MySQL读写分离

1.Amoeba服务器配置

(1)安装Java环境
cd /opt/
cp jdk-6u14-linux-x64.bin /usr/local/
cd /usr/local/
chmod +x jdk-6u14-linux-x64.bin
./jdk-6u14-linux-x64.bin
//按yes,按enter

mv jdk1.6.0_14/ /usr/local/jdk1.6
 
vim /etc/profile
  export JAVA_HOME=/usr/local/jdk1.6
  export CLASSPATH=.:$JAVA_HOME/lib:$JAVA_HOME/jre/lib
  export PATH=$JAVA_HOME/bin:$JAVA_HOME/jre/bin:$PATH
  export AMOEBA_HOME=/usr/local/amoeba
  export PATH=$PATH:$AMOEBA_HOME/bin
 
source /etc/profile
java -version

(2)安装Amoeba软件
mkdir /usr/local/amoeba
tar xf amoeba-mysql-binary-2.2.0.tar.gz -C /usr/local/amoeba/
chmod -R 755 /usr/local/amoeba/
/usr/local/amoeba/bin/amoeba
##如显示amoeba start|stop说明安装成功

(3)配置 Amoeba读写分离,两个Slave读负载均衡

三台服务器同样的命令

#先在Master、Slave1、Slave2 的mysql上开放权限给 Amoeba 访问
 
grant all on *.* to test@'192.168.88.%' identified by 'yy.com';

(4)配置amoeba服务
cd /usr/local/amoeba/conf/
cp amoeba.xml amoeba.xml.bak
vim amoeba.xml                   #修改amoeba配置文件
 
##30行##
<property name="user">amoeba</property>
##32行## 
<property name="password">123456</property>
##115行##
<property name="defaultPool">master</property>
##117-去掉注释-
<property name="writePool">master</property>
<property name="readPool">slaves</property>
 
cp dbServers.xml dbServers.xml.bak

vim dbServers.xml
#修改数据库配置文件
 
##23行##注释掉  作用:默认进入test库 以防mysql中没有test库时,会报错
<!-- <property name="schema">test</property> -->
##26##修改
<property name="user">test</property>
##28-30##去掉注释
<property name="password">123.com</property>
##45##修改,设置主服务器的名Master
<dbServer name="master"  parent="abstractServer">
##48##修改,设置主服务器的地址
<property name="ipAddress">192.168.80.10</property>
##52##修改,设置从服务器的名slave1
<dbServer name="slave1"  parent="abstractServer">
##55##修改,设置从服务器1的地址
<property name="ipAddress">192.168.80.11</property>
##58##复制上面6行粘贴,设置从服务器2的名slave2和地址
<dbServer name="slave2"  parent="abstractServer">
<property name="ipAddress">192.168.80.12</property>
##65行##修改
<dbServer name="slaves" virtual="true">
##71行##修改
<property name="poolNames">slave1,slave2</property>

/usr/local/amoeba/bin/amoeba start&	
#启动Amoeba软件,按ctrl+c 返回
netstat -anpt | grep java
#查看8066端口是否开启,默认端口为TCP 8066

2.测试读写分离

(1)主服务器上创建库和表,并添加数据

(2)在两台从服务器上查看表内容

(3)在从服务器上添加数据

(4)在主服务器上查看

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1363972.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

大学物理实验绪论——测量与误差

测量 测量的定义 以确定被测量对象量值为目的的操作称作测量。 测量的过程就是把被测物理量与选作计量单位的同类物理量进行比较的过程。 测量值的组成 物理量的测量值由数值和单位两部分组成。 测量的分类 测量分直接测量与间接测量。 等精度测量&#xff1a;对某一物理量多…

基于Java SSM框架实现实现机房预约系统项目【项目源码+论文说明】计算机毕业设计

基于java的SSM框架实现机房预约系统演示 摘要 21世纪的今天&#xff0c;随着社会的不断发展与进步&#xff0c;人们对于信息科学化的认识&#xff0c;已由低层次向高层次发展&#xff0c;由原来的感性认识向理性认识提高&#xff0c;管理工作的重要性已逐渐被人们所认识&#…

YOLOv8融合改进 更换检测头为Detect_DyHead同时添加C2f-EMSC和C2f-EMSCP模块

一、Detect_DyHead检测头和C2f-EMSC&#xff0c;C2f-EMSCP模块 详细介绍和代码在往期的博客里&#xff1a; Detect_DyHead&#xff1a; &#xff08;YOLOv8改进检测头Detect为Detect_Dyhead-CSDN博客&#xff09; C2f-EMSC和C2f-EMSCP&#xff1a; &#xff08;YOLOv8改进…

【SpringCloud Alibaba】Nacos服务管理与Feign负载均衡

目录 一、微服务搭建 1.1 服务提供者与服务消费者 1.2 依赖关系 二、服务注册与负载均衡使用 2.1 Nacos 实现服务的注册与发现 2.2 Loadbalancer负载均衡、Feign声明式服务调用 2.3 示例综合实现 2.3.1 服务注册与发现测试 2.3.2 负载均衡测试 一、微服务搭建 1.1 服…

Gen-AI 的知识图和分析(无需图数据库)

如今&#xff0c;图表比以往任何时候都更加相关和有用。由于目前正在发生的人工智能革命&#xff0c;工程师们正在考虑围绕 Gen-AI 的机会&#xff0c;利用具有动态提示、数据基础和屏蔽功能的开放 Gen-AI 解决方案&#xff0c;这进一步促使他们思考知识图谱等有效的解决方案。…

【动态规划】【滑动窗口】C++算法:100154 执行操作后的最大分割数量

作者推荐 【动态规划】【字符串】扰乱字符串 本文涉及的基础知识点 C算法&#xff1a;滑动窗口总结 动态规划 LeetCode100154 执行操作后的最大分割数量 给你一个下标从 0 开始的字符串 s 和一个整数 k。 你需要执行以下分割操作&#xff0c;直到字符串 s 变为 空&#xf…

【SpringCloud】之配置中心(进阶使用)

&#x1f389;&#x1f389;欢迎来到我的CSDN主页&#xff01;&#x1f389;&#x1f389; &#x1f3c5;我是君易--鑨&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; &#x1f31f;推荐给大家我的博客专栏《SpringCloud开发之远程消费》。&#x1f3af;&a…

C++ queue

目录 一、介绍 二、queue使用 三、模拟实现 四、优先级队列 五、priority_queue使用 OJ题&#xff1a;215. 数组中的第K个最大元素 快速排序 优先级队列 TOPK 六、模拟实现priority_queue 1、仿函数 2、优先级队列类 3、测试函数 一、介绍 1、队列是一种容器适配器…

阿里云服务器固定带宽实际下载速度表,不只是3M固定带宽

阿里云服务器公网带宽上传和下载速度对照表&#xff0c;1M带宽下载速度是128KB/秒&#xff0c;为什么不是1M/秒&#xff1f;阿里云服务器网aliyunfuwuqi.com分享阿里云服务器带宽1M、2M、3M、5M、6M、10M、20M、30M、50M、100M及200M等公网带宽下载速度对照表&#xff0c;附带宽…

VMware Workstation——修改虚拟机配置和设置网络

目录 一、修改配置 1、点击需要修改配置的虚拟机&#xff0c;然后点击编辑虚拟机配置 2、修改内存、CPU、硬盘配置 二、设置网络 1、从虚拟机配置中进入到网络适配器设置 2、选择网络连接模式 一、修改配置 1、点击需要修改配置的虚拟机&#xff0c;然后点击编辑虚拟机配…

Linux进程间通讯 -- 管道

Linux进程间通讯 – 管道 文章目录 Linux进程间通讯 -- 管道1. 原理2. 进程间通讯2.1 管道2.1.1 匿名管道 pipe2.2.2 有名管道 FIFO 2.2 信号2.3 共享内存2.4 本地套接字 1. 原理 Linux 进程间通讯&#xff0c;也称为IPC(InterProcess Communication) 在 Linux 中每个进程都具…

C++实现网站内搜索功能

文章目录 搜索结果的结构下载我们需要的数据分析html结构数据处理去标签之标题去标签之正文内容构造url把上述的数据清理操作对每一个文件都做一遍把处理好的数据都保存到一个.bin文件 构建正排索引构建倒排索引使用cpp-jieba分词计算每个文档中的每个词的权重对所有文档都进行…

格密码基础:光滑参数

目录 一. 铺垫高斯函数 二. 光滑参数图形理解 三. 光滑参数与格基本区 3.1 高斯与均匀分布的统计距离 3.2 光滑参数理解 四. 光滑参数与最短向量 五. 光滑参数与连续最小值 六. 光滑参数与对偶格的上界 七. 光滑参数与格的上界 八. 小结 一. 铺垫高斯函数 定义高斯密…

Django 9 常用通用视图分析

View 提供基于不同http方法执行不同逻辑的功能。 1. 创建 terminal输入 django-admin startapp the_13回车 2.tutorial子文件夹 settings.py注册一下 INSTALLED_APPS [django.contrib.admin,django.contrib.auth,django.contrib.contenttypes,django.contrib.sessions,dja…

九州金榜|孩子步入叛逆期,常常离家出走怎么办?

孩子在拥有了独立意识后&#xff0c;就开始试图挑战父母的权威。他们会主动去质疑父母&#xff0c;主动去证明自己的成熟和独立&#xff0c;还会主动试图逃离父母的控制范围。 近日就收到了家长求助孩子离家出走问题的私信&#xff0c;在得到家长同意&#xff0c;接下来我们就…

Docker-Compose部署Redis(v7.2)分片集群(含主从)

文章目录 一、前提准备1. 文件夹结构 二、配置文件1. redis.conf2. docker-compose文件 三、构建集群1. 自动分配主从关系2.1 构建3 master集群2.2 手动配置从节点 四、测试1. 集群结构2. 分片测试 环境 docker desktop for windows 4.23.0redis 7.2 目标 搭建如下图分片主从…

利用ArcGIS探究环境与生态因子对水体、土壤、大气污染物等影响的实践技术

如何利用ArcGIS实现电子地图可视化表达&#xff1f;如何利用ArcGIS分析空间数据&#xff1f;如何利用ArcGIS提升SCI论文的层次&#xff1f;制图是地理数据展现的直观形式&#xff0c;也是地理数据应用的必要基础。本次课程从ArcGIS的基本操作、ArcGIS 的空间数据分析及ArcGIS 的…

算法基础之货仓选址

货仓选址 核心思想&#xff1a; 贪心 绝对值不等式 : ∣ x – a ∣ ∣ x – b ∣ ≥ ∣ a – b ∣ |x – a| |x – b| ≥ |a – b| ∣x–a∣∣x–b∣≥∣a–b∣ 将n个数两两分组 1~~ n-1 (奇数会剩一个) 分别用绝对值不等式 即可推出来 货仓位置应该在中位数上(奇数) 或在中…

鸿蒙应用开发 闹钟实现

后台代理提醒简介 随着生活节奏的加快&#xff0c;我们有时会忘记一些重要的事情或日子&#xff0c;所以提醒功能必不可少。应用可能需要在指定的时刻&#xff0c;向用户发送一些业务提醒通知。例如购物类应用&#xff0c;希望在指定时间点提醒用户有优惠活动。为满足此类业务…

C# Unity将地形(Terrain)导出成obj文件

C# Unity将地形(Terrain)导出成obj文件 从其他地方搬运过来的&#xff0c;只能到出obj模型&#xff0c;不能导出贴图 using System.IO; using System.Text; using UnityEditor; using UnityEngine; using System;enum SaveFormat { Triangles, Quads } enum SaveResolution {…