mysql commit一个dml,在开启了binlog的情况下,为了保证binlog跟redo log 数据的一致性,mysql引入了两阶段提交。在事务commit的时候,会在两阶段提交的commit阶段,先将整个事务写入redolog,此时redolog还没有commit的标识。然后会将事务写入binlog,binlog commit成功之后,才会将commit标识写入redo log,并且告诉引擎层进行事务的提交。
以上的图片中可以看到,事务的提交主要分为两个主要步骤:
1. 准备阶段(Storage Engine(InnoDB) Transaction Prepare Phase)
此时SQL已经成功执行,并生成xid信息及redo和undo的内存日志。然后调用prepare方法完成第一阶段,papare方法实际上什么也没做,将事务状态设为TRX_PREPARED,并将redo log刷磁盘。
2. 提交阶段(Storage Engine(InnoDB)Commit Phase)
2.1 记录协调者日志,即Binlog日志。
如果事务涉及的所有存储引擎的prepare都执行成功,则调用TC_LOG_BINLOG::log_xid方法将SQL语句写到binlog(write()将binary log内存日志数据写入文件系统缓存,fsync()将binary log文件系统缓存日志数据永久写入磁盘)。
2.2 告诉引擎做commit。
最后,调用引擎的commit完成事务的提交。会清除undo信息,刷redo日志。
转载自:https://www.jianshu.com/p/b958722222b1