You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

13 KiB

10 | 走进黑盒SQL是如何在数据库中执行的

你好,我是李玥。

上一节课我们讲了怎么来避免写出慢SQL课后我给你留了一道思考题在下面这两个SQL中为什么第一个SQL在执行的时候无法命中索引呢

SELECT * FROM user WHERE left(department_code, 5) = '00028';
SELECT * FROM user WHERE department_code LIKE '00028%';

原因是这个SQL的WHERE条件中对department_code这个列做了一个left截取的计算对于表中的每一条数据都得先做截取计算然后判断截取后的值所以不得不做全表扫描。你在写SQL的时候尽量不要在WEHER条件中对列做任何计算。

到这里这个问题就结束了么那我再给你提一个问题这两个SQL中的WHERE条件虽然写法不一样但它俩的语义不就是一样的么是不是都可以解释成department_code这一列前5个字符是00028从语义上来说没有任何不同是吧所以它们的查询结果也是完全一样的。那凭什么第一条SQL就得全表扫描第二条SQL就可以命中索引

对于我们日常编写SQL的一些优化方法比如说我刚刚讲的“尽量不要在WEHER条件中对列做计算”很多同学只是知道这些方法但是却不知道为什么按照这些方法写出来的SQL就快

要回答这些问题需要了解一些数据库的实现原理。对很多开发者来说数据库就是个黑盒子你会写SQL会用数据库但不知道盒子里面到底是怎么一回事儿这样你只能机械地去记住别人告诉你的那些优化规则却不知道为什么要遵循这些规则也就谈不上灵活运用。

今天这节课我带你一起打开盒子看一看SQL是如何在数据库中执行的。

数据库是一个非常非常复杂的软件系统,我会尽量忽略复杂的细节,用简单的方式把最主要的原理讲给你。即使这样,这节课的内容仍然会非常的硬核,你要有所准备。

数据库的服务端,可以划分为执行器(Execution Engine)存储引擎(Storage Engine) 两部分。执行器负责解析SQL执行查询存储引擎负责保存数据。

SQL是如何在执行器中执行的

我们通过一个例子来看一下执行器是如何来解析执行一条SQL的。

SELECT u.id AS user_id, u.name AS user_name, o.id AS order_id
FROM users u INNER JOIN orders o ON u.id = o.user_id
WHERE u.id > 50

这个SQL语义是查询用户ID大于50的用户的所有订单这是很简单的一个联查需要查询users和orders两张表WHERE条件就是用户ID大于50。

数据库收到查询请求后需要先解析SQL语句把这一串文本解析成便于程序处理的结构化数据这就是一个通用的语法解析过程。跟编程语言的编译器编译时解析源代码的过程是完全一样的。如果是计算机专业的同学你上过的《编译原理》这门课其中很大的篇幅是在讲解这一块儿。没学过《编译原理》的同学也不用担心你暂时先不用搞清楚SQL文本是怎么转换成结构化数据的不妨碍你学习和理解这节课下面的内容。

转换后的结构化数据就是一棵树这个树的名字叫抽象语法树ASTAbstract Syntax Tree。上面这个SQL它的AST大概是这样的

这个树太复杂我只画了主要的部分你大致看一下能理解这个SQL的语法树长什么样就行了。执行器解析这个AST之后会生成一个逻辑执行计划。所谓的执行计划可以简单理解为如何一步一步地执行查询和计算最终得到执行结果的一个分步骤的计划。这个逻辑执行计划是这样的

LogicalProject(user_id=[$0], user_name=[$1], order_id=[$5])
    LogicalFilter(condition=[$0 > 50])
        LogicalJoin(condition=[$0 == $6], joinType=[inner])
            LogicalTableScan(table=[users])
            LogicalTableScan(table=[orders])

和SQL、AST不同的是这个逻辑执行计划已经很像可以执行的程序代码了。你看上面这个执行计划很像我们编程语言的函数调用栈外层的方法调用内层的方法。所以要理解这个执行计划得从内往外看。

  1. 最内层的2个LogicalTableScan的含义是把USERS和ORDERS这两个表的数据都读出来。
  2. 然后拿这两个表所有数据做一个LogicalJoinJOIN的条件就是第0列(u.id)等于第6列(o.user_id)。
  3. 然后再执行一个LogicalFilter过滤器过滤条件是第0列(u.id)大于50。
  4. 最后做一个LogicalProject投影只保留第0(user_id)、1(user_name)、5(order_id)三列。这里“投影(Project)”的意思是,把不需要的列过滤掉。

把这个逻辑执行计划翻译成代码然后按照顺序执行就可以正确地查询出数据了。但是按照上面那个执行计划需要执行2个全表扫描然后再把2个表的所有数据做一个JOIN操作这个性能是非常非常差的。

我们可以简单算一下如果user表有1,000条数据订单表里面有10,000条数据这个JOIN操作需要遍历的行数就是1,000 x 10,000 = 10,000,000行。可见这种从SQL的AST直译过来的逻辑执行计划一般性能都非常差所以需要对执行计划进行优化。

如何对执行计划进行优化,不同的数据库有不同的优化方法,这一块儿也是不同数据库性能有差距的主要原因之一。优化的总体思路是,在执行计划中,尽早地减少必须处理的数据量。也就是说,尽量在执行计划的最内层减少需要处理的数据量。看一下简单优化后的逻辑执行计划:

LogicalProject(user_id=[$0], user_name=[$1], order_id=[$5])
    LogicalJoin(condition=[$0 == $6], joinType=[inner])
        LogicalProject(id=[$0], name=[$1])              // 尽早执行投影
            LogicalFilter(condition=[$0 > 50])          // 尽早执行过滤
                LogicalTableScan(table=[users])
        LogicalProject(id=[$0], user_id=[$1])           // 尽早执行投影
            LogicalTableScan(table=[orders])

对比原始的逻辑执行计划,这里我们做了两点简单的优化:

  1. 尽早地执行投影,去除不需要的列;
  2. 尽早地执行数据过滤,去除不需要的行。

这样就可以在做JOIN之前把需要JOIN的数据尽量减少。这个优化后的执行计划显然会比原始的执行计划快很多。

到这里执行器只是在逻辑层面分析SQL优化查询的执行逻辑我们执行计划中操作的数据仍然是表、行和列。在数据库中表、行、列都是逻辑概念所以这个执行计划叫“逻辑执行计划”。执行查询接下来的部分就需要涉及到数据库的物理存储结构了。

SQL是如何在存储引擎中执行的

数据真正存储的时候,无论在磁盘里,还是在内存中,都没法直接存储这种带有行列的二维表。数据库中的二维表,实际上是怎么存储的呢?这就是存储引擎负责解决的问题,存储引擎主要功能就是把逻辑的表行列,用合适的物理存储结构保存到文件中。不同的数据库,它们的物理存储结构是完全不一样的,这也是各种数据库之间巨大性能差距的根本原因。

我们还是以MySQL为例来说一下它的物理存储结构。MySQL非常牛的一点是它在设计层面对存储引擎做了抽象它的存储引擎是可以替换的。它默认的存储引擎是InnoDB在InnoDB中数据表的物理存储结构是以主键为关键字的B+树每一行数据直接就保存在B+树的叶子节点上。比如上面的订单表组织成B+树,是这个样的:

这个树以订单表的主键orders.id为关键字组织其中“62:[row data]”表示的是订单号为62的一行订单数据。在InnoDB中表的索引也是以B+树的方式来存储的和存储数据的B+树的区别是,在索引树中,叶子节点保存的不是行数据,而是行的主键值。

如果通过索引来检索一条记录,需要先后查询索引树和数据树这两棵树:先在索引树中检索到行记录的主键值,然后再用主键值去数据树中去查找这一行数据。

简单了解了存储引擎的物理存储结构之后我们回过头来继续看SQL是怎么在存储引擎中继续执行的。优化后的逻辑执行计划将会被转换成物理执行计划物理执行计划是和数据的物理存储结构相关的。还是用InnoDB来举例直接将逻辑执行计划转换为物理执行计划

InnodbProject(user_id=[$0], user_name=[$1], order_id=[$5])
    InnodbJoin(condition=[$0 == $6], joinType=[inner])
        InnodbTreeNodesProject(id=[key], name=[data[1]])
            InnodbFilter(condition=[key > 50])
                InnodbTreeScanAll(tree=[users])
        InnodbTreeNodesProject(id=[key], user_id=[data[1]])
            InnodbTreeScanAll(tree=[orders])

物理执行计划同样可以根据数据的物理存储结构、是否存在索引以及数据多少等各种因素进行优化。这一块儿的优化规则同样是非常复杂的,比如,我们可以把对用户树的全树扫描再按照主键过滤这两个步骤,优化为对树的范围查找。

PhysicalProject(user_id=[$0], user_name=[$1], order_id=[$5])
    PhysicalJoin(condition=[$0 == $6], joinType=[inner])
        InnodbTreeNodesProject(id=[key], name=[data[1]])
            InnodbTreeRangeScan(tree=[users], range=[key > 50])  // 全树扫描再按照主键过滤,直接可以优化为对树的范围查找
        InnodbTreeNodesProject(id=[key], user_id=[data[1]])
            InnodbTreeScanAll(tree=[orders])

最终按照优化后的物理执行计划一步一步地去执行查找和计算就可以得到SQL的查询结果了。

理解数据库执行SQL的过程以及不同存储引擎中的数据和索引的物理存储结构对于正确使用和优化SQL非常有帮助。

比如我们知道了InnoDB的索引实现后就很容易明白为什么主键不能太长因为表的每个索引保存的都是主键的值过长的主键会导致每一个索引都很大。再比如我们了解了执行计划的优化过程后就很容易理解有的时候明明有索引却不能命中的原因是数据库在对物理执行计划优化的时候评估发现不走索引直接全表扫描是更优的选择。

回头再来看一下这节课开头的那两条SQL为什么一个不能命中索引一个能命中原因是InnoDB对物理执行计划进行优化的时候能识别LIKE这种过滤条件转换为对索引树的范围查找。而对第一条SQL这种写法优化规则就没那么“智能”了。

它并没有识别出来这个条件同样可以转换为对索引树的范围查找而走了全表扫描。并不是说第一个SQL写的不好而是数据库还不够智能。那现实如此我们能做的就是尽量了解数据库的脾气秉性按照它现有能力尽量写出它能优化好的SQL。

小结

一条SQL在数据库中执行首先SQL经过语法解析成AST然后AST转换为逻辑执行计划逻辑执行计划经过优化后转换为物理执行计划再经过物理执行计划优化后按照优化后的物理执行计划执行完成数据的查询。几乎所有的数据库都是由执行器存储引擎两部分组成,执行器负责执行计算,存储引擎负责保存数据。

掌握了查询的执行过程和数据库内部的组成你才能理解那些优化SQL的规则这些都有助于你更好理解数据库行为更高效地去使用数据库。

最后需要说明的一点是今天这节课所讲的内容不只是适用于我们用来举例的MySQL几乎所有支持SQL的数据库无论是传统的关系型数据库、还是NoSQL、NewSQL这些新兴的数据库无论是单机数据库还是分布式数据库比如HBase、Elasticsearch和SparkSQL等等这些数据库它们的实现原理也都符合我们今天这节课所讲的内容。

思考题

课后请你选一种你熟悉的非关系型数据库最好是支持SQL的当然不支持SQL有自己的查询语言也可以。比如说HBase、Redis或者MongoDB等等都可以尝试分析一下查询的执行过程对比一下它的执行器和存储引擎与MySQL有什么不同。

欢迎你在留言区与我讨论,如果你觉得今天的内容对你有帮助,也欢迎把它分享给你的朋友。