MySQL:从库复制半个事务会怎么样?

2022-06-17  本文已影响0人  轻松的鱼

复制异常

在复制过程中,主库发生故障或者网络中断,都会造成 slave io thread 中断,就有可能出现从库只复制了半个事务的情况。比如主库执行的事务如下:

begin;
insert 1;
insert 2;
commit;

从库接收的 binlog 可能只包含事务的一部分,比如:

从库的 slave sql thread 回放完这部分 binlog 后,会等待 slave io thread 从主库读取剩余的 binlog,而在此之前 sql 线程回放这半个事务,就和我们手工执行半个事务一样,不会提交也不会回滚。

我们应该如何应对这种异常呢?

实验过程

测试方法:

##1. 在从库上用 tc 模拟网络延迟,意在使读取 binlog 的速度变慢
tc qdisc add dev eth0 root netem delay 3000ms 3000ms 100%

##2. 在主库执行一个多语句事务
begin;
update t2 set pad='4' where id < 40;
update t2 set pad='5' where id < 50;
update t2 set pad='6' where id < 60;
commit;

##3. 在主库执行 commit 成功后,立刻用 iptables 切断主从之间的网络
iptables -A OUTPUT -d 172.16.21.4 -j DROP
iptables -A INPUT -s 172.16.21.4 -j DROP

这样我们可以在从库上观察到的现象为:

当 slave io thread 无法恢复

如果 slave io thread 长时间不能恢复,那么 sql 线程会因为等不到剩余的 binlog,一直无法提交或回滚,会一直持有这个事务的锁:

如果是主库故障导致的 slave io thread 异常,那很可能会进行主从切换,这个从库提升为主后,SQL线程持有的事务锁可能会阻塞业务请求。

此时应该 stop slave 停止 sql 线程,让事务回滚释放锁。需要注意的是:此情况下 stop slave 会等待 60 秒(等 slave io thread 接收事务剩余的binlog),60秒超时后才会停止 sql 线程:

当 slave io thread 恢复

slave io thread 异常中断后,sql 线程是正常工作的,sql 线程执行了部分事务,并且会等待 io 线程发送新的 binlog。slave io thread 线程恢复后,如果是基于 GTID 的复制,会从当前 GTID 事务开始重新获取完整的 binlog,从库会先回滚当前事务,然后再重新回放新收到的 binlog。

上一篇 下一篇

猜你喜欢

热点阅读