前言

学习一个新知识最好的方式就是上官网,所以我先把官网贴出来 MySQL官网 (点击查阅),如果大家有想了解我没有说到的东西可以直接上官网看哈~目前 MySQL 最新大版本为8.0,但是鉴于目前应用比较多的还是 5.7,所以今天在这里还是针对 5.7 来做讨论。
看了官网关于 MySQL 的介绍之后,我发现一个有趣的事情。在我身边的同事,很多都是把 MySQL 读错了,当然,也是因为大家已经约定俗成了,所以我卖的关子是,MySQL 大家一般会都成 my sequel,但是在官网上读法是这样的[ My Ess Que Ell ],即把 s q l 分开来读。当然这个不重要啦,这里只是跟大家唠嗑一下哈哈~想验证的伙伴可以点击这个What is MySQL?

下面开始进入正题:
下面是 MySQL 的发展过程,目前的系统基本上都是分布式微服务的了,由于支持事务的特性,所以 innodb 为默认的存储引擎,也是我们今天课程的主角。(MyISAM 和 Innodb 的区别在此不做赘述,想了解 MySQL 的引擎可至 MySQL 引擎链接查阅。

这里有一张脑图,想要完整高清图片可以到微信我的公众号下【6曦轩】下回复 MySQL 脑图获取:

接下来我们会以这张脑图的一些知识点展开来讲,但是由于文章篇幅有限,有些点可能只会一笔带过,有兴趣的小伙伴可以到我的公众号下与我留言讨论。
我们今天的重点,在于将 MySQL 语句的执行流程给大家梳理一遍(如果文章哪里有疏漏的话,尽请大家批评指正)。

正文

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

查询语句的执行分为以下几步:

  1. 查询缓存
  2. 解析器生成解析树
  3. 预处理再次生成解析树
  4. 查询优化器
  5. 查询执行计划
  6. 查询执行引擎
  7. 查询数据返回结果

查询缓存
通过如下语句可查看缓存开关情况(默认关闭):show variables like 'query_cache%';

  1. MySQL 拿到一个查询请求后先会在查询缓存中看看是否执行过此语句,之前执行的语句会以 key-value 的形式缓存在内存中,key 是缓存的语句,value 是查询的结果
  2. 如果命中缓存则直接将结果返回,如果没有命中则继续执行后面

在 MySQL 中默认是关闭的,官方也建议关闭,将缓存交托给第三方如 redis 处理,为啥:

  1. 查询缓存的失效特表频繁,对一个表的更新都会失效这个表所有的查询缓存,对于更新频繁的表命中率太低
  2. MySQL 8.0 直接删除查询缓存

解析器生成解析树

  1. 语法解析

语法解析是解析你的语句是不是满足 MySQL 语法标准,如果不对则会 :
ERROR 1064 (42000): You have an error in your SQL syntax … 关于错误码在官网有说明

  1. 词法解析

关于解析完生成的解析树类似下图,我以’select name from user_info where sex=1 and age>20’为例:

预处理再次生成解析树
语义解析,在语法及词法解析完之后,进行预处理之后再次生成解析树。
查询优化器
在这一步将前面生成的解析树优化成一个执行计划。
在这步做的事情主要有:

  1. 选择最合适的索引;
  2. 选择表扫还是走索引;
  3. 选择表关联顺序;
  4. 优化 where 子句;
  5. 排除管理中无用表;
  6. 决定 order by 和 group by 是否走索引;
  7. 尝试使用 inner join 替换 outer join;
  8. 简化子查询,决定结果缓存;
  9. 合并试图;

顺便提一下,optimizer_trace 优化器追踪器,在 MySQL 中是默认关闭的(毕竟开启也会消耗性能嘛对吧),可以使用 set 语句修改一下 optimizer_trace的开关,感受一下:set optimizer_trace='enabled=on '
先查询优化器追踪的开关:show variables like 'optimizer_trace%';

执行完一条语句之后执行下面语句查看优化器追踪:select * from information_schema.optimizer_traceG
可以看到一个 json 类型的字符串,主要是语句优化的三个阶段,篇幅有限,这里不展开,对照着看应该可以看懂。

查询执行计划
查询最后一次查询的消耗,用以比较开销:show status like 'Last_query_cost';
在这一步选择开销最小的计划执行
查询执行引擎
这里执行器会先对权限做一个判断,如果有权限,才会执行以下步骤,否则跑出权限异常:

  1. 调用 Innodb 引擎接口获取这个表的第一行,判断ID是否为10,如果不是跳过,如果是则存在结果集中;
  2. 引擎执行下一行,重复判断相同的逻辑,直到最后一行;
  3. 最后将满足结果的结果集返回;
  4. 对于有索引的表也差不多,第一次是调用满足结果的第一行接口, 下来是查找满足结果的下一行接口

查询数据返回结果
将查询数据的结果返回给查询的客户端,如果有缓存则返回缓存(前面已经说了默认关闭),可以说就大功告成了哈哈哈哈,真是曲折。
总结
经过上面一系列的梳理,相信大家对 MySQL 查询语句的流程也有了一个大致的了解,下面是针对查询语句的流程做的一张图,方便大家记忆理解:

By the way

有问题?可以给我留言或私聊
有收获?那就顺手点个赞呗~
当然,也可以到我的公众号下「6曦轩」,
回复“学习”,即可领取一份【Java工程师进阶架构师的视频教程】~
回复“面试”,可以获得:【本人呕心沥血整理的 Java 面试题】
回复“MySQL脑图”,可以获得【MySQL 知识点梳理高清脑图】
由于我咧,科班出身的程序员,php,Android以及硬件方面都做过,不过最后还是选择专注于做 Java,所以有啥问题可以到公众号提问讨论(技术情感倾诉都可以哈哈哈),看到的话会尽快回复,希望可以跟大家共同学习进步,关于服务端架构,Java 核心知识解析,职业生涯,面试总结等文章会不定期坚持推送输出,欢迎大家关注~~~

http://weixin.qq.com/r/2kM-J4DEIN1frcRE9xbI (二维码自动识别)

mysql 查询语句_MySQL相关(一)- 一条查询语句是如何执行的相关推荐

  1. mysql直连1.执行语句_MySQL随笔01_一条SQL语句是如何执行的

    一.MySQL基础架构示意图 二.MySQL分层 总体来讲,MySQL可以分为二层:Server层 和 存储引擎 两部分,如下图所示. 不同的存储引擎共用一个Server层. 三.各部分组件概述 连接 ...

  2. mysql语句统计总数_一条sql语句实现统计查询_MySQL

    bitsCN.com 一条sql语句实现统计查询 如图:程序员在进行如下的统计时,现在提供两种实现方案: 方案一:运用 SEKECT CASE WHEN EXPLAIN SELECT count(*) ...

  3. mysql查询数量语句_mysql语句统计总数_一条sql语句实现统计查询_MySQL

    bitsCN.com 一条sql语句实现统计查询 如图:程序员在进行如下的统计时,现在提供两种实现方案: 方案一:运用 SEKECT CASE WHEN EXPLAIN SELECT count(*) ...

  4. mysql上一条语句成功_mysql : 获取上一条insert语句

    在一些项目中 , 经常接触分表 . 比如 : 商品信息 和 商品的详情 , 是分开的两个表 . dt_mall和dt_mall_content; 当我dt_mall插入一条数据的时候 , 如果插入成功 ...

  5. mysql关联查询去重_MySQL外键和高级查询(连接查询、联合查询、子查询、去重查询)...

    MySQL的外键 什么是外键,很简单保持数据一致性的一个约束键.如果你有两张表,第一张是学生表,第二张表是一个成绩表,我们来看看保持数据一致性,其实在Django等框架的模型中中也能做关联获取对象. ...

  6. mysql 架构优化_Mysql 架构及优化之-查询性能优化

    ①②③④⑤⑥⑦⑧⑨ 查询执行基础知识 mysql执行查询过程 ① 客户端将查询发送到服务器 ② 服务器检查查询缓存 如果找到了就从缓存返回结果 否则进行下一步 ③ 服务器解析,预处理和优化查询,生成执 ...

  7. mysql select from user_select * from user 这条 SQL 语句,背后藏着哪些不可告人的秘密?...

    作为一名 Java开发人员,写 SQL 语句是常有的事,但是你知道 SQL 语句背后的处理逻辑吗?比如下面这条 SQL 语句: select * from user where id=1 执行完这条语 ...

  8. mysql 慢查询优化_MySQL 性能优化之慢查询

    性能优化的思路 首先需要使用慢查询功能,去获取所有查询时间比较长的SQL语句 其次使用explain命令去查询由问题的SQL的执行计划(脑补链接:点我直达1,点我直达2) 最后可以使用show pro ...

  9. mysql命令行语句_MySql命令行命令和SQL语句

    一.常用mysql命令行命令 1.启动MYSQL服务 net start mysql 停止MYSQL服务 net stop mysql 2.netstat -na|findstr 3306 查看被监听 ...

最新文章

  1. Spring Boot项目快速搭建
  2. pyhton 安装pip 以及 numpy (解决python:ModuleNotFoundError:No module named numpy 等类似缺包问题
  3. springboot中的过滤器、拦截器、监听器整合使用
  4. Java队列 PriorityQueue
  5. mysql 计算gps坐标距离_mysql、sqlserver和php计算GPS经纬度坐标距离
  6. UICollectionView,CollectionView,瀑布流
  7. 小姐姐亲身体验:在阿里数据库科研团队实习是种怎样的体验?
  8. 一个感染型的病毒逆向分析
  9. Ajax入门(创建 XMLHttpRequest 对象)
  10. 单模光电转换器怎么接_以太网光纤收发器怎么用?
  11. 如何解决NLP分类任务的11个关键问题:类别不平衡低耗时计算小样本鲁棒性测试检验长文本分类 JayLou娄杰
  12. PHP学习笔记【13】_正则表达式
  13. 千脑——您的在线电脑
  14. sublime开发apicloud项更新apploader解决
  15. php10天速成培训,十天学会php之第十天
  16. 深度分析《英雄联盟》游戏运营商背后的大数据支撑体系
  17. nmake的调用方法
  18. 《ShowYou数组代码》第38题:对数组元素进行排序
  19. Tensorflow Serving部署模型
  20. EKL日志平台架构概括

热门文章

  1. 复合机 涂布机_涂布复合机适用的范围在那些地方?
  2. 对于Java回调的最深刻解析
  3. 阿里开源Euler:国内首个工业级图表征学习框架
  4. CentOS7下安装GUI图形界面
  5. Android Gallery和ImageSwitcher同步自动(滚动)播放图片库
  6. [TODO]Python拾遗(二)
  7. openstack运维实战系列(十七)之glance与ceph结合
  8. META http-equiv=X-UA-Compatible content=IE=EmulateIE7
  9. HNOI2008 玩具装箱
  10. ASP.NET 2.0 中的代码隐藏和编译