我们还是从一个表的一条更新语句说起,我们创建下面一张表: create table T(ID int primary key, c int); 如果要将ID=2这一行c的值加1,sql可以这么写: update T set
我们还是从一个表的一条更新语句说起,我们创建下面一张表:
create table T(ID int primary key, c int);
如果要将ID=2这一行c的值加1,sql可以这么写:
update T set c=c+1 where ID=2;
前一篇文章介绍过SQL语句基本的执行链路,可以确认的说,查询语句的那一套流程,更新语句也是同样会走一遍。在执行语句前要先连接数据库,这是连接器的工作。接下来,分析器会通过词法和语法解析知道这是一条更新语句,优化器决定要使用ID这个索引。然后执行器负责具体执行,找到这一行,然后更新。
与查询流程不一样的是,更新流程还涉及两个重要的日志模块:redo log(重做日志)和binlog(归档日志)。
Mysql里面经常说到的WAL技术的全称是Write-Ahead Logging,它的关键点就是行写日志再写磁盘。具体来说,当有一条记录需要更新的时候,InnoDB引擎会先把记录写到redo log里面,并更新内存,这个时候更新就算完成了。同时InnoDB引擎会在适当的时候,将这个操作记录更新到磁盘里面,而这个更新往往是在系统比较闲的时候做。
InnoDB的redo log是固定在小的,比如可以配置为一组4个文件,每个文件的大小是1GB,那么总共就可以记录4GB的操作,如图所示:
从头开始写,写到末尾又回到开头循环写,write pos是当前记录的位置,一边写一边后移,写到3号文件末尾后就回到0号文件开头。checkpoint是当前要擦除的位置,擦出记录前要把记录更新到数据文件中。如果write pos追上checkpoint,这个时候不能再执行新的更新,得停下来先擦除一些记录,把checkpoint推进一下。
有了redo log,InnoDB就可以保证即使在数据库发生异常重启,之前提交的记录都不会丢失,这个能力称为crash-safe。
mysql整体来看,其实就两块:一块是Server层,它主要做的是MySQL功能层面的事情;还有一块就是引擎层,负面存储相关的具体事宜,上面说到的redo log是InnoDB引擎特有的日志,而Server层也有自己的日志,称为binlog。
为什么会有两份日志呢?因为最开始MySQL并没有InnoDb引擎,MySQL自带的引擎是MyISAM,但MyISAM并没有crash-safe能力,binlog日志只能用于归档。
redo log日志和binlog日志有以下3个不同点:
那么执行一个简单的update语句的内部流程如下:
将redo log的写入折成了两个步骤:prepare和commit,这就是“两阶段提交”。
简单来说,redo log和binlog都可以用于表示事务提交状态,而两阶段提交就是让这两个状态保持逻辑上的一致。
--结束END--
本文标题: 02.日志系统:一条SQL更新语句是如何执行的?
本文链接: https://www.lsjlt.com/news/2366.html(转载时请注明来源链接)
有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341
下载Word文档到电脑,方便收藏和打印~
2024-05-16
2024-05-16
2024-05-16
2024-05-15
2024-05-15
2024-05-15
2024-05-15
2024-05-15
2024-05-15
2024-05-15
回答
回答
回答
回答
回答
回答
回答
回答
回答
回答
0