热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

MySQL8.0官方文档第八章优化(十一)——外连接简化

目录第八章优化(十一)——外连接简化8.2优化SQL语句8.2.1优化SELECT语句8.2.1.10外连接简化第八章优化(十一ÿ

目录

  • 第八章 优化(十一)—— 外连接简化

    • 8.2 优化SQL语句
      • 8.2.1 优化 SELECT 语句
        • 8.2.1.10 外连接简化


第八章 优化(十一)—— 外连接简化


8.2 优化SQL语句


8.2.1 优化 SELECT 语句


8.2.1.10 外连接简化

查询语句中,FROM子句的表与表连接表达式在很多情况下都能得到简化。

在解析器处理阶段,右外连接操作的查询都会被转换为仅包含左外连接操作的等效查询。通常,都会对这样的右连接执行如下的等效转换:

(T1, ...) RIGHT JOIN (T2, ...) ON P(T1, ..., T2, ...)

变成以下等效的左连接:

(T2, ...) LEFT JOIN (T1, ...) ON P(T1, ..., T2, ...)

所有的内连接表达式,例如:T1 INNER JOIN T2 ON P(T1,T2)形式都会替换为 T1,T2 列表,而 P(T1,T2)作为把所有的WHERE的条件(或嵌套连接的连接条件,如果有的话)连接后的表现形式。(译者:这句话的意思是用 FROM T1, T2 ON P(T1, T2)表示表的内连接和WHERE条件)

当优化器安排表之间的连接操作计划时,它只考虑以下计划:对于每个外连接操作,外部表都会先于内部表被访问。优化器的选择是有限的,因为只有这样的计划才允许使用嵌套循环算法执行外连接。

考虑以下查询,其中 R(T2) 条件会极大地减小表T2中匹配行的数量:

SELECT * FROM T1 LEFT JOIN T2 ON P1(T1, T2) # 译者:官方文档原文中漏写了关键字FROMWHERE P(T1, T2) AND R(T2)

如果按照上面书写的查询顺序执行,那么优化器就别无选择,只能先访问限制较少的表T1,再访问限制较多的表T2,这可能会导致非常低效的执行计划。

相反,如果WHERE条件是空拒绝的(null-rejected),那么,MySQL会将该查询转换为没有外连接操作的查询(也就是说,它将外连接转换为内连接)。如果一个条件对于由连接操作生成的任何空补行(NULL-complemented row)的测试结果都为FALSE或UNKNOWN(不知道),则该条件被称为外连接操作的空拒绝条件。(译者:这里的意思是:如果连接操作会生成一些空补行,而这些空补行在WHERE条件测试中结果一定是假,就不能实际生成空补行,因此,这种外连接的结果与内连接是一样的。)

因此,对于以下外连接:

T1 LEFT JOIN T2 ON T1.A = T2.A

如果该外连接添加以下任意条件,该外连接就会转成内连接。这是因为这些条件都是空拒绝的,它们对于所有空补行(空补行是由于表T2的所有列都设置为NULL造成的)都不可能是真:

T2.B IS NOT NULL
T2.B > 3
T2.C <&#61; T1.C
T2.B < 2 OR T2.C > 1

而以下这些条件就不是空拒绝的&#xff0c;因为它们对于空补行可能测试为真&#xff1a;

T2.B IS NULL # 如果T2.B是NULL&#xff0c;则该表达时为真
T1.B < 3 OR T2.B IS NOT NULL # 由于T1.B <3 可以是真&#xff0c;可以不管T2如何&#xff0c;整个表达式为真
T1.B < 3 OR T2.B > 3

检查外联接操作的条件是否为空拒绝的基本规则很简单&#xff1a;

  • A IS NOT NULL的这种形式&#xff0c;其中 A 是任何内部表的属性

  • 是包含对内部表引用的谓词&#xff0c;当其参数之一为NULL时&#xff0c;该谓词的测试结果为未知&#xff08;UNKNOWN&#xff09;

  • 是一个在连接操作时出现空拒绝条件的连词

  • 是空拒绝条件的拆分形式&#xff08;disjunction&#xff09;

查询中一个外连接操作中的条件可以是空拒绝的&#xff0c;而另一个连接操作的条件是不空拒绝的。在以下查询中&#xff0c;第二个外连接操作中的WHERE条件是空拒绝的&#xff0c;而第一个不是。

SELECT * FROM T1 LEFT JOIN T2 ON T2.A &#61; T1.ALEFT JOIN T3 ON T3.B &#61; T1.BWHERE T3.C > 0 # 该条件是空拒绝的

对于原始查询&#xff0c;优化器只评估与单个表 T1、T2、T3 这样的访问顺序兼容的执行计划。但是&#xff0c;对于重写的查询&#xff0c;它还可考虑为 T3、T1、T2 这样的访问顺序。

一个外连接操作的转换可以触发另一个外连接操作的转换。因此&#xff0c;该查询&#xff1a;

SELECT * FROM T1 LEFT JOIN T2 ON T2.A &#61; T1.ALEFT JOIN T3 ON T3.B &#61; T2.B # 译者不太明白为何可以把T1.B变成T2.B&#xff1f;&#xff08;与上面语句对比&#xff09;WHERE T3.C > 0

首先转换为以下查询&#xff1a;&#xff08;译者&#xff1a;原文说&#xff0c;这是第一转换&#xff0c;说明上面第二个连接条件本来是&#xff1a;T3.B &#61; T2.B&#xff0c;所以&#xff0c;以上两条查询语句本身是没有关系的。&#xff09;

SELECT * FROM T1 LEFT JOIN T2 ON T2.A &#61; T1.AINNER JOIN T3 ON T3.B &#61; T2.B # 由于T3表是空拒绝的&#xff0c;所以把左连接转换为内连接WHERE T3.C > 0

第二次转换&#xff0c;等效于以下查询&#xff1a;

SELECT * FROM (T1 LEFT JOIN T2 ON T2.A &#61; T1.A), T3WHERE T3.C > 0 AND T3.B &#61; T2.B

因为条件T3.B &#61; T2.B是空拒绝的&#xff0c;所以留下的外连接操作也可转换为内连接。这将导致最终转换为一个完全没有外部联接的查询&#xff1a;

SELECT * FROM (T1 INNER JOIN T2 ON T2.A &#61; T1.A), T3WHERE T3.C > 0 AND T3.B &#61; T2.B

有时&#xff0c;优化器能成功地替换内嵌的外连接操作&#xff0c;但无法转换嵌入的外连接。看以下查询&#xff1a;

SELECT * FROM T1 LEFT JOIN(T2 LEFT JOIN T3 ON T3.B &#61; T2.B)ON T2.A &#61; T1.AWHERE T3.C > 0

转换为&#xff1a;

SELECT * FROM T1 LEFT JOIN(T2 INNER JOIN T3 ON T3.B &#61; T2.B)ON T2.A &#61; T1.AWHERE T3.C > 0

只能重写为仍然包含内嵌的外连接操作的形式&#xff1a;

SELECT * FROM T1 LEFT JOIN(T2,T3)ON (T2.A &#61; T1.A AND T3.B &#61; T2.B)WHERE T3.C > 0

在查询中&#xff0c;任何尝试转换内嵌外连接的操作都必须考虑内嵌外连接的连接条件和WHERE条件。在以下查询中&#xff0c;对于嵌入的外连接&#xff0c;WHERE条件不是空拒绝的&#xff0c;但是&#xff0c;嵌入外连接的连接条件 T2.A &#61; T1.A AND T3.C &#61; T1.C是空拒绝的&#xff1a;

SELECT * FROM T1 LEFT JOIN(T2 LEFT JOIN T3 ON T3.B &#61; T2.B)ON T2.A &#61; T1.A AND T3.C &#61; T1.C # 应该是 T3.C &#61; T1.C 条件可能为假&#xff0c;导致整个条件为假。WHERE T3.D > 0 OR T1.D > 0 # T1.D > 0 可以为真&#xff0c;导致WHERE子句为真&#xff0c;所以不是空拒绝的。

因此&#xff0c;该查询可以转换为&#xff1a;

SELECT * FROM T1 LEFT JOIN(T2, T3)ON T2.A &#61; T1.A AND T3.C &#61; T1.C AND T3.B &#61; T2.BWHERE T3.D > 0 OR T1.D > 0

上一集 MySQL 8.0 官方文档 第八章 优化&#xff08;十&#xff09;—— 外连接优化

下一集 MySQL 8.0 官方文档 第八章 优化&#xff08;十二&#xff09;—— 多范围读优化


推荐阅读
  • MyBatis多表查询与动态SQL使用
    本文介绍了MyBatis多表查询与动态SQL的使用方法,包括一对一查询和一对多查询。同时还介绍了动态SQL的使用,包括if标签、trim标签、where标签、set标签和foreach标签的用法。文章还提供了相关的配置信息和示例代码。 ... [详细]
  • 本文介绍了如何在MySQL中将零值替换为先前的非零值的方法,包括使用内联查询和更新查询。同时还提供了选择正确值的方法。 ... [详细]
  • 本文介绍了如何使用php限制数据库插入的条数并显示每次插入数据库之间的数据数目,以及避免重复提交的方法。同时还介绍了如何限制某一个数据库用户的并发连接数,以及设置数据库的连接数和连接超时时间的方法。最后提供了一些关于浏览器在线用户数和数据库连接数量比例的参考值。 ... [详细]
  • ubuntu用sqoop将数据从hive导入mysql时,命令: ... [详细]
  • 本文详细介绍了如何使用MySQL来显示SQL语句的执行时间,并通过MySQL Query Profiler获取CPU和内存使用量以及系统锁和表锁的时间。同时介绍了效能分析的三种方法:瓶颈分析、工作负载分析和基于比率的分析。 ... [详细]
  • 本文介绍了在开发Android新闻App时,搭建本地服务器的步骤。通过使用XAMPP软件,可以一键式搭建起开发环境,包括Apache、MySQL、PHP、PERL。在本地服务器上新建数据库和表,并设置相应的属性。最后,给出了创建new表的SQL语句。这个教程适合初学者参考。 ... [详细]
  • 在数据分析工作中,我们通常会遇到这样的问题,一个业务部门由若干业务组构成,需要筛选出每个业务组里业绩前N名的业务员。这其实是一个分组排序的 ... [详细]
  • 使用Ubuntu中的Python获取浏览器历史记录原文: ... [详细]
  • 图解redis的持久化存储机制RDB和AOF的原理和优缺点
    本文通过图解的方式介绍了redis的持久化存储机制RDB和AOF的原理和优缺点。RDB是将redis内存中的数据保存为快照文件,恢复速度较快但不支持拉链式快照。AOF是将操作日志保存到磁盘,实时存储数据但恢复速度较慢。文章详细分析了两种机制的优缺点,帮助读者更好地理解redis的持久化存储策略。 ... [详细]
  • Oracle分析函数first_value()和last_value()的用法及原理
    本文介绍了Oracle分析函数first_value()和last_value()的用法和原理,以及在查询销售记录日期和部门中的应用。通过示例和解释,详细说明了first_value()和last_value()的功能和不同之处。同时,对于last_value()的结果出现不一样的情况进行了解释,并提供了理解last_value()默认统计范围的方法。该文对于使用Oracle分析函数的开发人员和数据库管理员具有参考价值。 ... [详细]
  • 本文介绍了一个在线急等问题解决方法,即如何统计数据库中某个字段下的所有数据,并将结果显示在文本框里。作者提到了自己是一个菜鸟,希望能够得到帮助。作者使用的是ACCESS数据库,并且给出了一个例子,希望得到的结果是560。作者还提到自己已经尝试了使用"select sum(字段2) from 表名"的语句,得到的结果是650,但不知道如何得到560。希望能够得到解决方案。 ... [详细]
  • 高质量SQL书写的30条建议
    本文提供了30条关于优化SQL的建议,包括避免使用select *,使用具体字段,以及使用limit 1等。这些建议是基于实际开发经验总结出来的,旨在帮助读者优化SQL查询。 ... [详细]
  • 本文讨论了在数据库打开和关闭状态下,重新命名或移动数据文件和日志文件的情况。针对性能和维护原因,需要将数据库文件移动到不同的磁盘上或重新分配到新的磁盘上的情况,以及在操作系统级别移动或重命名数据文件但未在数据库层进行重命名导致报错的情况。通过三个方面进行讨论。 ... [详细]
  • 前景:当UI一个查询条件为多项选择,或录入多个条件的时候,比如查询所有名称里面包含以下动态条件,需要模糊查询里面每一项时比如是这样一个数组条件:newstring[]{兴业银行, ... [详细]
  • MySQL外键1对多问题的解决方法及实例
    本文介绍了解决MySQL外键1对多问题的方法,通过准备数据、创建表和设置外键关联等步骤,实现了用户分组和插入数据的功能。详细介绍了数据准备的过程和外键关联的设置,以及插入数据的示例。 ... [详细]
author-avatar
沈巛小糖meimei昌策_247
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有