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

约定不等于承诺〃 2022-01-22 23:15 452阅读 0赞

大家六一儿童节好呀!

接下来的一段时间内,将带领大家一同探索MySQL的奥妙,加油吧!我们。

下面进入正题:一条SQL语句是如何进行的?

对于这个问题,我想将其分为两个问题来回答,分别是:

  • 一条查询SQL是如何执行的?
  • 一条更新SQL是如何执行的?

    我们都知道MySQL内部是分为Server层和存储引擎层的;每个层都有各自的职责;

watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dmeDE1NTAyMTA0MTEy_size_16_color_FFFFFF_t_70

对于一个查询语句,eg:select * from T where ID=10

第一步:连接器(负责跟客户端简历连接、获取权限、维持和管理连接)

  1. mysql -h IP -p Prot -u User -p Password #密码错误会提醒“Access denied for user”

如果验证通过,连接器会到表里面查出你拥有的权限;这个连接里面的权限判断逻辑,都依赖于此权限。

假如一个用户连接成功后,即使管理员账号对这个用户的权限进行了修改,也不会影响当前连接,除非重新建立连接。

完成后,如果你没有后续的动作,这个连接就处于空闲状态,可用“show processlist”命令查看。显示sleep则为空闲

这里涉及到一个长连接和短连接

  • 长连接:建立连接成功后,如果客户端持续有请求,则一直使用同一个连接。

    • 会发现有些时候MySQL占用内存涨的特别快,这是因为MySQL在执行过程中临时使用的内存是管理在连接对象中的,连接断开即释放。现象举例:MySQL异常重启

      • 方案一:定期断开长连接
      • 方案二:每次执行一个比较大的操作后,执行”mysql_reset_connection”(此过程不会重新做权限验证)
  • 短连接:执行完很少的几次查询就断开连接

第二步:查询缓存

  • 之前执行过的查询,MySQL以”Key - Value”的形式存在内存(key为SQL,value为结果集);
  • 只要对该表有一个更新,则这个表上的查询缓存都会被清空;
  • 手动创建命令:select SQL_CACHE * from T(MySQL8中已经彻底废除此功能);

第三步:分析器(词法分析 —— 语法分析)

词法分析:通过“select”,识别出为查询;通过“T”,识别出表名;通过条件“ID”,识别出ID那一列;等等;

语法分析:如果语法有误,则提示“You have an error in your SQL syntax”;

第四步:优化器

  1. 决定用哪个索引;联查表连接顺序;条件执行优先级 ,等等;

第五步:执行器 (执行SQL)

第六步:存储引擎(提供读写接口,供执行器调用并获取结果集)

  • 首先会判断你是否有该权限;
  • 如果命中查询缓存,则会在返回结果的时候进行权限验证;

补充:MySQL慢查询日志中,会有一个rows_examined字段,代表扫描了多少行。这个值就是在每次调用存储引擎获取数据行的时候累加的。不过,若是只调用一次,而扫描了多行,则rows_examined是不准确的。

对于一条更新SQL,同样走和查询一样的流程。

不过,执行更新SQL会涉及到两个重要的日志:redo log (重做日志)、bin log (归档日志)。

“《孔乙己》中,酒店掌柜有一个粉板,专门用来记录客人的赊账记录,如果赊账的人不多,那么他可以把顾客名和账目写在粉板上。但如果赊账太多了,粉板总有挤不下的时候,这个时候,掌柜一定还有一个专门记账的账本”

掌柜第一种做法:每次记账都打开账本

掌柜第二种做法:先写在粉板上,打烊后再核算

个人感觉掌柜很聪明。对于MySQL,同样也有类似的处理:

  • MySQL的 WAL技术,(Write-Ahead-Logging),先写日志,再写磁盘。
  • 对于innodb,会先把记录写到redo log中,并更新内存,同时,innodb会在适当的时候,将这个操作记录更新到磁盘中
  • 有了redo log 会保证innodb在数据库发生异常重启之后,之前提交的记录都不会丢失,称之为:“crash_safe”

    redo log是存储引擎层的日志,bin log是server层的日志






















redo log bin log
innodb独有 server层日志,所有引擎都有
物理日志,记录“在某个数据页上做了什么修改” 逻辑日志,记录的是语句的原始逻辑
循环写,空间固定 追加写入

以上就是执行一个SQL语句会经历的过程,当中提到的很多名词,后期会慢慢给出对应详细解释,大家也可以提出来。

努力吧!坚持做正确的事!相信自己!!!

发表评论

表情:
评论列表 (有 0 条评论,452人围观)

还没有评论,来说两句吧...

相关阅读

    相关 SQL语句如何执行

    平时我们使用数据库,看到的通常都是一个整体。我们看到的只是输入一条语句,返回一个结果,却不知道这条语句在 MySQL 内部的执行过程。为了弄清楚一条SQL语句是如何执行的,必须