MySQL的主从复制

MySQL的主从复制

#主从复制介绍
主从复制基于binlog实现的
主库发生新的操作,都会记录binlog
从库取得主库的binlog进行回放
主从复制的过程是异步的

主从复制的前提

(1) 2个或以上的数据库实例
(2) 主库需要开启二进制日志
(3) server_id要不同,区分不同的节点
(4) 主库需要建立专用的复制用户
(5) 从库应该通过备份主库,恢复的方法进行数据恢复
(6) 告诉从库一些复制信息(ip port user pass,二进制日志起点)
(7) 从库应该开启专门的复制线程

主从复制搭建

准备多实例
[root@mysql ~]# systemctl start mysqld3307
[root@mysql ~]# mysqld --initialize-insecure --user=mysql --basedir=/application/mysql --datadir=/data/3308/data
[root@mysql ~]# systemctl start mysqld3308
[root@mysql ~]# mysql -S /data/3308/mysql.sock -e "select @@port"
+--------+
| @@port |
+--------+
|   3308 |
+--------+
[root@mysql ~]# mysql -S /data/3307/mysql.sock -e "select @@port"
[root@mysql ~]# mysql -uroot -p123456 -S /data/3307/mysql.sock -e "select @@port"
mysql: [Warning] Using a password on the command line interface can be insecure.
+--------+
| @@port |
+--------+
|   3307 |
+--------+
[root@mysql ~]# 
检查配置文件
[root@mysql ~]# cat /data/3307/my.cnf 
[mysqld]
basedir=/application/mysql
datadir=/data/3307/data
socket=/data/3307/mysql.sock
log_error=/data/3307/mysql.log
port=3307
server_id=7
log_bin=/data/3307/mysql-bin

[root@mysql ~]# cat /data/3308/my.cnf 
[mysqld]
basedir=/application/mysql
datadir=/data/3308/data
socket=/data/3308/mysql.sock
log_error=/data/3308/mysql.log
port=3308
server_id=8
log_bin=/data/3308/mysql-bin
[root@mysql ~]# 

主库创建复制用户
[root@mysql ~]# mysql -uroot -p123456 -S /data/3307/mysql.sock -e "grant replication slave on *.* to ‘repl‘@‘10.0.1.%‘ identified by ‘123456‘"
通过主库进行数据恢复
#主库
[root@mysql ~]# mysqldump -uroot -p123456 -S /data/3307/mysql.sock -A --master-data=2 --single-transaction -R -E --triggers >/mnt/full.sql

#从库
[root@mysql ~]# mysql -S /data/3308/mysql.sock 
mysql> set sql_log_bin=0;
mysql> source /mnt/full.sql
mysql> set sql_log_bin=1;
配置从库信息
[root@mysql ~]# mysql -S /data/3308/mysql.sock 
mysql> CHANGE MASTER TO
    -> MASTER_HOST=‘10.0.1.110‘,
    -> MASTER_USER=‘repl‘,
    -> MASTER_PASSWORD=‘123456‘,
    -> MASTER_PORT=3307,
    -> MASTER_LOG_FILE=‘mysql-bin.000007‘,
    -> MASTER_LOG_POS=444,
    -> MASTER_CONNECT_RETRY=10;

从库开启复制线程
[root@mysql ~]# mysql -S /data/3308/mysql.sock 
mysql> start slave;

检查主从复制状态
[root@mysql ~]# mysql -S /data/3308/mysql.sock 
mysql> show slave status \G
             Slave_IO_Running: Yes
            Slave_SQL_Running: Yes

主从复制原理

#主从复制中涉及的文件
主库: 
	binlog 
从库: 
	relaylog  中继日志
	master.info  主库信息文件
	relaylog.info relaylog应用的信息

#主从复制中涉及的线程
主库:
	Binlog_Dump Thread : DUMP_T
从库: 
	SLAVE_IO_THREAD     : IO_T
	SLAVE_SQL_THREAD    : SQL_T

#主从复制工作(过程)原理
(1) 从库 执行change master to语句 ,将以下信息保存至master.info文件中
(2) 从库 start slave , 从库IO_T和SQL_T生成
(3)Io_T ,读取master.info中主库连接信息(ip , port, user,password)
   连接主库,主库专门开启Dump_T,和从库IO_T进行交互
(4) IO_T,读取 master.info ,复制的起点信息,找主库DUMP要最新的binlog
(5)主库dump_T,截取最新的binlog,发送给从库IO_T
(6)扩展: 基于TCP/IP网络工作模式,IO_T将日志存储到TCP/Ip缓存中,并返回ACK给主库.
(7) IO_T最终会将缓存中数据,写入到relay_log文件中保存.更新master.info为新位置点.
(8) SQL_T,读取relay-log.info信息,获取上次SQL_T回放的位置点.
(9)SQL_T,根据位置点,向下回放最新的relaylog.并且再次更新relay-log.info文件为最新点.
补充:
(10) 主库dump_T 会持续的监控binlog的变化,一旦有新的日志生成,给从库发一些信号.
(11) 从库 relay_log_purge线程会定期自动清理回放过的relay日志.

主从复制监控

mysql> show slave status \G

#主库有关的信息(master.info):
Master_Host: 10.0.0.51
Master_User: repl
Master_Port: 3307
Connect_Retry: 10
Master_Log_File: mysql-bin.000004
Read_Master_Log_Pos: 609

#从库relay应用信息有关的(relay.info):
Relay_Log_File: db01-relay-bin.000002
Relay_Log_Pos: 320
Relay_Master_Log_File: mysql-bin.000004

#从库线程运行状态(排错)
Slave_IO_Running: Yes
Slave_SQL_Running: Yes
Last_IO_Errno: 0
Last_IO_Error: 
Last_SQL_Errno: 0
Last_SQL_Error:

#过滤复制有关的信息:
Replicate_Do_DB: 
Replicate_Ignore_DB: 
Replicate_Do_Table: 
Replicate_Ignore_Table: 
Replicate_Wild_Do_Table: 
Replicate_Wild_Ignore_Table: 
 
#从库延时主库的时间(秒):
Seconds_Behind_Master: 0

#延时从库:
SQL_Delay: 0
SQL_Remaining_Delay: NULL

#GTID复制有关的状态信息
Retrieved_Gtid_Set: 
Executed_Gtid_Set: 
Auto_Position: 0

主从复制故障

#IO 线程故障 
(1) 连接主库 
网络不通,防火墙
连接信息错误(ip,port,user,password)
解决: 
1. stop slave 
2. reset slave all;
3. change master to 
4. start slave

#请求Binlog
主库日志损坏 ,丢失,不连续

模拟故障:
主库 reset master 处理

解决:
从库 
stop slave ;
reset slave all; 
CHANGE MASTER TO 
MASTER_HOST=‘10.0.1.110‘,
MASTER_USER=‘repl‘,
MASTER_PASSWORD=‘123456‘,
MASTER_PORT=3307,
MASTER_LOG_FILE=‘mysql-bin.000001‘,
MASTER_LOG_POS=154,
MASTER_CONNECT_RETRY=10;
start slave;

#SQL线程故障
回放日志 (为什么SQL语句执行失败?)
(1) 版本不一样(SQL_Mode)
(2) 要创建的对象已存在 (主从数据不一致)
(3) 要删除或修改的对象不存在(主从数据不一致)
(5) 约束冲突(主键)

处理建议: 
1. 一切以主库为准.最彻底方法重新构建主从。
2. 将从库设置为只读库,采用读写分离的中间件,防止从库写入
mysql> show variables like ‘%read_only%‘;
read_only=ON
super_read_only=ON

主从延时监控及原因

5.6.1 主库方面原因
#binlog写入不及时
解决:
sync_binlog=1	#每次事务提交时,一定保证binlog写入磁盘

#传统复制
默认情况下dump_t是串行传输binlog
在并发事务量大时或者大事务,由于dump_t 是串型工作的,导致传送日志较慢
MySQL 为了解决这个问题,5.6版本支持了GTID复制,使得dump_t可以并行传输事务.
5.7版本之后,加强了GTID功能,建议都开启GTID

#其他原因
网络慢
主机配置
主从参数不一致
从库较多

#从库方面原因
传统复制中如果主库并发事务量很大,或者出现大事务
由于从库是单SQL线程,导致,不管传的日志有多少,只能一次执行一个事务.
5.6 版本,有了GTID,可以实现多SQL线程,但是只能基于不同库的事务进行并发回放.(database) 
5.7 版本中,增强了GTID,增加了seq_no,增加了新型的并发SQL线程模式(logical_clock),MTS技术
主从延时监控
#主从延时的监控
mysql> show slave status \G
        Seconds_Behind_Master: 0
#主库方面监控
mysql> show master status;
+------------------+----------+--------------+------------------+-------------------+
| File             | Position | Binlog_Do_DB | Binlog_Ignore_DB | Executed_Gtid_Set |
+------------------+----------+--------------+------------------+-------------------+
| mysql-bin.000007 |      609 |              |                  |                   |
+------------------+----------+--------------+------------------+-------------------+
1 row in set (0.00 sec)

mysql> 

[root@mysql data]# cat relay-log.info 
7
./db01-relay-bin.000002
953
mysql3307-bin.000001
21797376
0
0
1

[root@db01 data]# 

MySQL的主从复制

上一篇:MySQL-优化


下一篇:MySQL日志管理