MySQL 核心模块揭秘 | 07 期 | 二阶段提交 (1) prepare 阶段
作者:操盛春,爱可生技术专家,公众号『一树一溪』作者,专注于研究 MySQL 和 OceanBase 源码。爱可生开源社区出品,原创内容未经授权不得随意使用,转载请联系小编并注明来源。
本文基于 MySQL 8.0.32 源码,存储引擎为 InnoDB。
目录
1. 二阶段提交
2. prepare 阶段
2.1 binlog prepare
2.2 InnoDB prepare
3. 总结
正文
1. 二阶段提交
二阶段提交,顾名思义,包含两个阶段,它们是:
- prepare 阶段。
- commit 阶段。
我们只考虑 SQL 语句操作 InnoDB 表的场景,对于用户事务,是否使用二阶段提交,取决于是否开启了 binlog。
因为 MySQL 把 binlog 也看作一个存储引擎,开启 binlog,SQL 语句改变(插入、更新、删除)InnoDB 表的数据,这个 SQL 语句执行过程中,就涉及到两个存储引擎。
使用二阶段提交,就是为了保证两个存储引擎的数据一致性。
用户事务提交分为两种场景,如果开启了 binlog,它们都会使用二阶段提交。
场景 1:通过 BEGIN 或其它开始事务的语句,显式开始一个事务,用户手动执行 COMMIT 语句提交事务。
场景 2:没有显式开始的事务,一条 SQL 语句执行时,InnoDB 会隐式开始一个事务,SQL 语句执行完成之后,自动提交事务。
如果没有开启 binlog,SQL 语句改变表中数据,不产生 binlog,不用保证 binlog 和表中数据的一致性,用户事务也就不需要使用二阶段提交了。
InnoDB 内部事务是个特例,不管是否开启了 binlog,改变表中数据都不会产生 binlog 日志,所以内部事务不需要使用二阶段提交。
2. prepare 阶段
以下代码中,ha_prepare_low() 会调用 binlog 和 InnoDB 处理 prepare 逻辑的方法。
int MYSQL_BIN_LOG::prepare(THD *thd, bool all) {<br> ...<br> thd->durability_property = HA_IGNORE_DURABILITY;<br> ...<br> int error = ha_prepare_low(thd, all);<br> ...<br>}<br>