我的编程空间,编程开发者的网络收藏夹
学习永远不晚

02.日志系统:一条SQL更新语句是如何执行的?

短信预约 信息系统项目管理师 报名、考试、查分时间动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

02.日志系统:一条SQL更新语句是如何执行的?

   我们还是从一个表的一条更新语句说起,我们创建下面一张表:

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(归档日志)。

  • redo log

  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。  

  • binlog

  MySQL整体来看,其实就两块:一块是Server层,它主要做的是MySQL功能层面的事情;还有一块就是引擎层,负面存储相关的具体事宜,上面说到的redo log是InnoDB引擎特有的日志,而Server层也有自己的日志,称为binlog。

 为什么会有两份日志呢?因为最开始MySQL并没有InnoDb引擎,MySQL自带的引擎是MyISAM,但MyISAM并没有crash-safe能力,binlog日志只能用于归档。

redo log日志和binlog日志有以下3个不同点:

  1. redo log 是InnoDB引擎特有的;binlog 是MySQL的Server层实现的,所有引擎都可以使用。
  2. redo log 是物理日志,记录的是“在某个数据页上做了什么修改”;binlog是逻辑日志,记录的是这个语句的原始逻辑,比如"给ID=2这一行的c字段+1"。
  3. redo log 是循环写的,空间固定会用完;binlog是可以追加写入的,“追加写”是指binlog文件写到一定的大小后会切换到下一个,并不会覆盖以前的日志。

那么执行一个简单的update语句的内部流程如下:

  1. 执行器先找引擎取ID=2这一行,ID是主键,引擎直接用树搜索找到这一行,如果ID=2这一行所在的数据页本来就在内存中,就直接返回给执行器;否则,需要先从磁盘读入内存,然后返回。
  2. 执行器拿到引擎返回的行数据,把这个值+1,行到新的一行数据,再调用引擎接口写入这行新数据。
  3. 引擎将这行数据更新到内存中,同时将这个更新操作记录到redo log里面,此时redo log处理prepare状态,然后告知执行器执行完了,随时可以提交事务。
  4. 执行器生成这条操作的binlog,并把binlog写入磁盘。
  5. 执行器调用引擎的提交事务接口,引擎把刚刚写入redo log改成提交(commit)状态,更新完成。

将redo log的写入折成了两个步骤:prepare和commit,这就是“两阶段提交”。

  • 两阶段提交
  1. 先写redo log 后写binlog。假设在redo log写完后,binlog还没有写完的时候,MySQL进程异常重启,根据redo log恢复回来后这一行数据c=1,但由于binlog没有写完就crash了,这个时候binglog里面就没有这条更新语句,如果需要用这个binlog来恢复临时库的话,这个临时库就会少一次更新,恢复出来的数据c=0。
  2. 先写binlog后写redo log。如果binlog写完之后crash,由于redo log还没有写完,崩溃恢复后这个事务无效,所以这一行的c=0,但时,binlog里面已经记录了c=1,所以之后用binlog恢复出来的数据c=1写原库的值不同。

简单来说,redo log和binlog都可以用于表示事务提交状态,而两阶段提交就是让这两个状态保持逻辑上的一致。

 

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

02.日志系统:一条SQL更新语句是如何执行的?

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

02 | 日志系统:一条SQL更新语句是如何执行的?

极客时间专栏《MySQL实战45讲》的阅读笔记。
02 | 日志系统:一条SQL更新语句是如何执行的?
2015-04-30

02.日志系统:一条SQL更新语句是如何执行的?

我们还是从一个表的一条更新语句说起,我们创建下面一张表: create table T(ID int primary key, c int);    如果要将ID=2这一行c的值加1,SQL可以这么写: update T set c=c+1 where ID=
02.日志系统:一条SQL更新语句是如何执行的?
2015-02-19

一条SQL更新语句是如何执行的

文章首发于公众号「蝉沐风」,认真写好每一篇文章,欢迎大家关注交流这是图解MySQL的第2篇文章,这篇文章会通过一条SQL更新语句的执行流程让大家清楚地明白:什么是InnoDB页?缓存页又是什么?为什么这么设计?什么是表空间?不同存储引擎的表在文件系统的底层表示
一条SQL更新语句是如何执行的
2014-08-03

一条SQL更新语句的执行过程是什么

这篇“一条SQL更新语句的执行过程是什么”文章的知识点大部分人都不太理解,所以小编给大家总结了以下内容,内容详细,步骤清晰,具有一定的借鉴价值,希望大家阅读完这篇文章能有所收获,下面我们一起来看看这篇“一条SQL更新语句的执行过程是什么”文
2023-06-30

一条SQL查询语句是如何执行的?

导读Mysql在中小型企业中是个香饽饽,目前主流的数据库之一,几乎没有一个后端开发者不会使用的,但是作为一个老司机,仅仅会用真的不够。今天陈某透过一个简单的查询语句来讲述在Mysql内部的执行过程。select * from table where id=10
一条SQL查询语句是如何执行的?
2019-06-13

分析mysql中一条SQL查询语句是如何执行的

目录一、MySQL 逻辑架构概览二、连接器(Connector)三、查询缓存(Query Cache)四、解析器(Parser)五、优化器(Optimizer)六、执行器七、小结一、MySQL 逻辑架构概览 MySQL 最重要、最与众不同的
2022-05-30

一条简单的更新语句,MySQL是如何加锁的?

看如下一条sql语句:# table T (id int, name varchar(20))delete from T where id = 10;MySQL在执行的过程中,是如何加锁呢?在看下面这条语句:select * from T where id =
一条简单的更新语句,MySQL是如何加锁的?
2020-02-11

编程热搜

目录