119. MySQL主从复制

主从复制基础 (!=同步,异步的工作模式)

1. 主从复制介绍及原理(Master-Slave Replication)

两台以上的数据库实例,通过二进制日志实现数据复制关系.
在这里插入图片描述
1.1 架构划分
一主一从
在这里插入图片描述
一主多从
在这里插入图片描述

2. 主从复制作用

辅助数据备份.比较擅长处理数据库的物理损坏.
架构演变: 高可用,读写分离,分布式…

3. 主从复制前提 (Classic Replication搭建过程)

(1) 两台以上数据库实例,需要不同的server_id ,server_uuid保持不同
(2) 主库需要开启二进制日志(binlog),专用复制用户
(3) 进行主库数据,恢复到从库.
(4) 从库:change master to , 通知从库,主库:user,password (专用的),ip,port,复制的起点.
(5) 从库:start slave; 开启专用的复制线程

4. 主从复制搭建

4.1 节点准备及各项检查

systemctl start mysqld3307
systemctl start mysqld3308

mysql -S /tmp/mysql3307.sock  -e "select @@server_id"
mysql -S /tmp/mysql3308.sock  -e "select @@server_id"
mysql -S /tmp/mysql3307.sock  -e "select @@log_bin"
mysql -S /tmp/mysql3307.sock  -e "select @@server_uuid"
mysql -S /tmp/mysql3308.sock  -e "select @@server_uuid"

4.2 binlog日志检查及用户准备

mkdir -p /data/mysql/binlog_3307

chown -R mysql.mysql /data 

vim /data/mysql/my3307.cnf 
添加以下行:
log_bin=/data/mysql/binlog_3307/mysql-bin          
                               
systemctl restart mysqld3307
mysql -S /tmp/mysql3307.sock  -e "grant replication slave on *.* to repl@'10.0.0.%' identified by '123'"

4.3 备份主库数据,恢复到从库,并记录位置点.

mysqldump -S /tmp/mysql3307.sock -A --master-data=2 --single-transaction -R -E --triggers >/tmp/full.sql
过滤出以下这行数据
-- CHANGE MASTER TO MASTER_LOG_FILE='mysql-bin.000001', MASTER_LOG_POS=444;
mysql -S /tmp/mysql3308.sock < /tmp/full.sql

4.4 从库执行 change master to

CHANGE MASTER TO
  MASTER_HOST='10.0.0.51',
  MASTER_USER='repl',
  MASTER_PASSWORD='123',
  MASTER_PORT=3307,
  MASTER_LOG_FILE='mysql-bin.000001',
  MASTER_LOG_POS=444,
  MASTER_CONNECT_RETRY=10;

4.5 开启复制线程(IO,SQL)

start slave;
mysql -S /tmp/mysql3308.sock -e "show slave status\G"|grep Running:

4.6 简单诊断问题思路:

(1) 线程 
IO 
SQL

(2) 看具体报错 
last_xx_error 

(3) 处理 
stop slave 
change master to 
start slave

4.7 主从复制原理

文件:  
	主库: binlog 
	从库: relaylog,master.info,relay-log.info 
线程:
	主库: binlog dump thread(show processlist;)
	从库: IO  , SQL

简单原理介绍:
要使用主从复制,需开启二进制日志功能。主服务器有数据更新,会写入二进制日志,由slave服务线程通过网络传给从服务器,从服务器使用io线程来处理写入从服务器的中继日志,SQL线程从中继日志获取写到数据库中。

4.8 主从复制的监控方式

show slave status \G
(1) 主库连接信息 
Master_Host: 10.0.0.51
Master_User: repl
Master_Port: 3307
Master_Log_File: mysql-bin.000001
Read_Master_Log_Pos: 444

(2) 从库中继日志回放位置点
Relay_Log_File: db01-relay-bin.000002
Relay_Log_Pos: 320
Exec_Master_Log_Pos: 444
Seconds_Behind_Master: 0

说明: 主要是为了定位主从延时问题.


(3) 从库复制线程状态
Slave_IO_Running: Yes
Slave_SQL_Running: Yes

Last_IO_Errno: 0
Last_IO_Error: 
Last_SQL_Errno: 0
Last_SQL_Error: 	
	
说明: 必须是yes , 但是yes不代表完全没问题.		
	
	
(4) 过滤复制相关状态
Replicate_Do_DB: 
Replicate_Ignore_DB: 
Replicate_Do_Table: 
Replicate_Ignore_Table: 
Replicate_Wild_Do_Table: 
Replicate_Wild_Ignore_Table: 


(5) 延时从库
SQL_Delay: 0
SQL_Remaining_Delay: NULL

说明: 作用是用来处理逻辑故障.


(6) GTID复制有关:
Retrieved_Gtid_Set: 
Executed_Gtid_Set: 

4.9 主从复制故障分析及处理

(1) 线程为什么不工作
IO线程 :  
	1. 连接主库
	网络: ip , port , 防火墙 ,user,password  , 网络不通.
	解决方案: 保证以上信息正确可用.
	stop slave 
	change master to 
	start slave
	
	2. 请求日志
	日志位置点指定错误(搭建过程问题)
	binlog损坏,误删等.
	例如: reset master;
    保险的处理方法:重新备份恢复,重新构建主从
		
	3. 存储日志
	relaylog 损坏,丢失,不连续.


SQL线程:
1. 回放relaylog  ----> SQL ----> SQL线程执行SQL语句为什么失败?
(1) 创建对象失败,删除修改的对象不存在.
从库被写入了.
解决方案: 
方法一:
stop slave; 
set global sql_slave_skip_counter = 1;
#将同步指针向下移动一个,如果多次不同步,可以重复操作。
start slave;
方法二:
/etc/my.cnf
slave-skip-errors = 1032,1062,1007
常见错误代码:
1007:对象已存在
1032:无法执行DML
1062:主键冲突,或约束冲突

终极大招: 
从库只读
read_only=on
super_read_only=on	 
或者,配合中间件.		 
		 		 
		  
(2) 版本不一致,SQL_mode不一致,配置不一致.
解决方案: 尽量统一. 将高版本的配置降低为低版本配置.


(3) DML语句执行失败
异步复制会导致的问题,比如:表不存在.
人工校验主从的数据不一致原因,把有问题的操作给补上.
 
(4) 约束冲突,例如: 主键自增列冲突
双主模式下,会导致主键冲突.
主库宕机,8.0以前自增列是没有持久化的.
 
扩展: 
pt-checksum
pt-sync 
pt-heartbeat

发布了148 篇原创文章 · 获赞 65 · 访问量 7623

猜你喜欢

转载自blog.csdn.net/chengyinwu/article/details/103810095