MySQL 语句执行过程详解

MySQL 原理篇html

MySQL 索引机制mysql

MySQL 体系结构及存储引擎linux

MySQL 语句执行过程详解sql

MySQL 执行计划详解数据库

MySQL InnoDB 缓冲池缓存

MySQL InnoDB 事务服务器

MySQL InnoDB 锁并发

MySQL InnoDB MVCCapp

MySQL InnoDB 实现高并发原理函数

MySQL InnoDB 快照读在RR和RC下有何差别

当客户端向 MySQL 发送一个请求的时候,MySQL 的执行过程以下图所示:

MySQL 客户端/服务端通讯

通讯机制

MySQL 客户端与服务端的通讯方式是 “ 半双工 ”。

  • 全双工:双向通讯,发送同时也能够接收
  • 半双工:双向通讯,同时只能接收或者是发送,没法同时作操做
  • 单工:只能单一方向传送

一旦一端开始发送消息,另外一端要接收完整个消息才能响应它,因此咱们没法也无须将一个消息切成小块独立发送,也没有办法进行流量控制。

客户端用一个单独的数据包将查询请求发送给服务器,因此当查询语句很长的时候,须要设置 max_allowed_packet 参数。

可是须要注意的是,若是查询实在是太大,服务端会拒绝接收更多数据并抛出异常。

与之相反的是,服务器响应给用户的数据一般会不少,由多个数据包组成。可是当服务器响应客户端请求时,客户端必须完整的接收整个返回结果,而不能简单的只取前面几条结果,而后让服务器中止发送。

于是在实际开发中,尽可能保持查询简单且只返回必需的数据,减少通讯间数据包的大小和数量是一个很是好的习惯,这也是查询中尽可能避免使用 SELECT * 以及加上 LIMIT 限制的缘由之一。

链接状态

对于一个 MySQL 的链接,或者说一个线程,时刻都有一个状态来标识这个链接正在作什么。

能够经过以下命令来查看链接的状态:

show full processlist
show processlist

详细的状态集描述参考官网:https://dev.mysql.com/doc/refman/5.7/en/general-thread-states.html

这里简单介绍几个经常使用的链接状态:

  • Sleep:线程正在等待客户端发送数据
  • Query:链接线程正在执行查询
  • Locked:线程正在等待表锁的释放
  • Sorting result:线程正在对结果进行排序
  • Sending data:向请求端返回数据

对于出现问题的链接能够经过 kill {id} 的方式进行杀掉。

查询缓存

工做原理:

  • 缓存 SELECT 操做的结果集和 SQL 语句。
  • 新的 SELECT 语句,先去查询缓存,判断是否存在可用的记录集,须要注意的是在判断的时候,要求 SQL 语句彻底同样(SQL 两端容许存在空格)才会匹配到缓存数据。

缓存参数

MySQL 的缓存参数在配置文件中设置,能够经过以下命令来查看缓存的参数:

show variables like 'query_cache%'

  1. query_cache_type
    1. 0:不启用查询缓存,默认值
    2. 1:启用查询缓存,只要符合查询缓存的要求,客户端的查询语句和记录集均可以缓存起来,供其余客户端使用,SQL 语句中加上 SQL_NO_CACHE 将不缓存
    3. 2:启用查询缓存,只要查询语句中添加了参数:SQL_CACHE,且符合查询缓存的要求,客户端的查询语句和记录集,则能够缓存起来,供其余客户端使用
  1. query_cache_size
    1. 总的缓存池的大小,容许设置 query_cache_size 的值最小为40K,默认1M,推荐设置为64M/128M
    2. 当总的缓存池大小超过设置的值时,会按照时间顺序,让最老的缓存失效
  1. query_cache_limit
    1. 指定单个查询可以使用的缓冲区大小,默认设置为1M

缓存执行状况

能够经过以下命令来查看缓存状况:

show status like 'Qcache%'

  1. Qcache_free_blocks
    1. Query Cache 中目前还有多少剩余的 blocks。若是该值显示较大,则说明 Query Cache 中的内存碎片较多了,可能须要寻找合适的机会进行整理
  1. Qcache_free_memory
    1. Query Cache 中目前剩余的内存大小。经过这个参数咱们能够较为准确的观察出当前系统中的Query Cache 内存大小是否足够,是须要增长仍是过多了
  1. Qcache_hits
    1. 缓存命中次数。经过这个参数咱们能够查看到 Query Cache 的基本效果
  1. Qcache_inserts
    1. 插入缓存的记录数,经过 Qcache_hits 和 Qcache_inserts 两个参数咱们就能够算出 Query Cache 的命中率,Query Cache 命中率 = Qcache_hits / ( Qcache_hits + Qcache_inserts )
  1. Qcache_lowmem_prunes
    1. 多少条 Query 由于内存不足而被清除出 Query Cache。经过 Qcache_lowmem_prunes 和 Qcache_free_memory 相互结合,可以更清楚的了解到咱们系统中 Query Cache 的内存大小是否真的足够,是否常常出现由于内存不足而有 Query 被清除
  1. Qcache_not_cached
    1. 由于 query_cache_type 的设置或者不能被 cache 的 Query 的数量
  1. Qcache_queries_in_cache
    1. 当前 Query Cache 中 cache 的 Query 数量
  1. Qcache_total_blocks
    1. 当前 Query Cache 中的 block 数量

不会缓存的状况

  1. 当查询语句中设置了 SQL_NO_CACHE,则不会被缓存。
  2. 当查询语句中有一些不肯定的数据时,则不会被缓存。如包含函数 NOW() ,CURRENT_DATE() 等相似的函数,或者用户自定义的函数,存储函数,用户变量等都不会被缓存。
  3. 当查询的结果大于 query_cache_limit 设置的值时,结果不会被缓存。
  4. 对于 InnoDB 引擎来讲,当一个语句在事务中修改了某个表,那么在这个事务提交以前,全部与这个表相关的查询都没法被缓存。所以长时间执行事务,会大大下降缓存命中率。
  5. 查询的表是系统表。
  6. 查询语句不涉及到表。

缓存有哪些坑?

  1. 在查询以前必须先检查是否命中缓存,浪费计算资源。
  1. 若是这个查询能够被缓存,那么执行完成后,MySQL 发现查询缓存中没有这个查询,则会将结果存入查询缓存,这会带来额外的系统消耗。
  1. 针对表进行写入或更新数据时,将对应表的全部缓存都设置失效。
  1. 若是查询缓存很大或者碎片不少时,这个操做可能带来很大的系统消耗。

适用场景

以读为主的业务,数据生成 以后就不常改变的业务,好比门户类、新闻类、报表类、论坛类。

查询优化处理

查询优化处理的三个阶段

  1. 解析 SQL
    1. 经过 lex 语法分析,yacc 语法分析将 SQL 语句解析成解析树。
    2. lex、yacc 语法参考:https://www.ibm.com/developerworks/cn/linux/sdk/lex/
  1. 预处理阶段
    1. 根据 MySQL 的语法的规则进一步检查解析树的合法性,如:检查数据的表和列是否存在,解析名字和别名的设置。还会进行权限的验证
  1. 查询优化器
    1. 优化器的主要做用就是找到最优的执行计划

查询优化器如何找到最优执行计划

这里介绍几种优化方式,更多的能够参考《高性能MySQL_第3版(中文)》

  1. 使用等价变化规则
    1. 5 = 5 and a > 5 改写成 a > 5
    2. a < b and a = 5 改写成 b > 5 and a = 5
    3. 基于联合索引,调整条件位置等
  1. 优化 count、min、max 等函数
    1. InnoDB 引擎 min 函数只需找索引最左边
    2. InnoDB 引擎 max 函数只需找索引最右边
    3. MyISAM 引擎 count(*),不须要计算,直接返回
  1. 覆盖索引扫描
  1. 子查询优化
    1. select * from (select * from user where id = 1) as t;,会被优化成一级查询
  1. 提早终止查询
    1. 用了 limit 关键字或者使用不存在的条件,获取到 limit 所须要的数据后,就再也不遍历接下来的数据
  1. IN 的优化
    1. MySQL 对于 IN 的查询,会先进性排序,再采用二分查找的方式查找数据
    2. 好比表中的数据是 1,2,3,4,5,where 条件是 id IN(2,1,3),在进行 IN 操做的时候,会先对 IN 中的数据排序,变成 1,2,3,而后取出一条数据1先和2比较,1<2,则往2的左边查找,进而找到1,接下来就是再获取一条数据重复上面的查找步骤。
    3. 其余关系型数据库不会采用二分查找的方式,而是和 or 的方式同样,where id=1 or id=2 or id=3,从表中获取一条数据和 where 条件中的 or 的数据一个一个比对。

MySQL 的查询优化器是基于成本计算的原则,它会尝试各类执行计划,数据抽样的方式进行试验(随机的读取一个 4K 的数据块进行分析)。

执行计划

这块内容比较多,后面会单独提供一篇文章描述

查询执行引擎

调用插件式的存储引擎的原子 API 进行执行计划的执行。

返回客户端

  1. 有须要作缓存的,执行缓存操做
  2. 增量的返回执行结果,开始生成第一条结果时,MySQL 就开始往请求方逐步返回数据,这样作的好处是 MySQL 服务器无须保存过多的数据,浪费内存,用户体验好,立刻就拿到了数据

参考

http://www.sohu.com/a/233335431_468739

相关文章
相关标签/搜索