开发同事问,为什么一个标量子查询,放在where子句后进行大小判断,比不放在where子句后进行判断大小运行的更快?按道理加了一次判断,不是应该变慢么?

把语句拿过来,看了一下两个语句的执行计划:

语句1和执行计划1:

SELECT A.*,
/*剩余量*/
(A.q_basic -
(SELECT COALESCE(SUM(d.q_basic), 0.00)
FROM e_order d
WHERE CAST(d.r_item AS INT) = A.ID
AND d.req_status NOT IN ('FZ'))) AS surplus
FROM e_order A
LEFT JOIN e_requirement b
ON A.requirement_no = b.requirement_no
LEFT JOIN erp_project C
ON b.factory = C.project_no
WHERE 1 = 1
AND A.status IN ('WC')
AND (A.q_basic -
(SELECT COALESCE(SUM(d.q_basic), 0.00)
FROM e_order d
WHERE CAST(d.r_item AS INT) = A.ID
AND d.req_status NOT IN ('FZ'))) > 0.00
AND (C.project_name LIKE CONCAT('%', 'csg18098', '%') OR
C.project_no LIKE CONCAT('%', 'csg18098', '%'))
AND A.requirement_no LIKE CONCAT('%', '0000004390', '%');

语句2和执行计划2:

SELECT A.*,
/*剩余量*/
(A.q_basic -
(SELECT COALESCE(SUM(d.q_basic), 0.00)
FROM e_order d
WHERE CAST(d.r_item AS INT) = A.ID
AND d.req_status NOT IN ('FZ'))) AS surplus
FROM e_order A
LEFT JOIN e_requirement b
ON A.requirement_no = b.requirement_no
LEFT JOIN erp_project C
ON b.factory = C.project_no
WHERE 1 = 1
AND A.status IN ('WC')
AND (C.project_name LIKE CONCAT('%', 'csg18098', '%') OR
C.project_no LIKE CONCAT('%', 'csg18098', '%'))
AND A.requirement_no LIKE CONCAT('%', '0000004390', '%')

从上面的执行计划看,在where之后进行大小判断后,执行时间是662.954 ms;去掉判断后执行时间是1549.644 ms。的确如开发所说。

现在分别来看上面的两个执行计划。
语句1在where子句后增加判断,表关联的顺序是((((a,d_1),b),c),d)。语句2不在where子句后加判断的关联顺序是(((a,b),c),d)。

其实这里d_1就是表示在where子句后的表e_order。这一点,可以将语句修改一下,就可以得到验证:

SELECT A.*,
/*剩余量*/
(A.q_basic -
(SELECT COALESCE(SUM(d.q_basic), 0.00)
FROM e_order d
WHERE CAST(d.r_item AS INT) = A.ID
AND d.req_status NOT IN ('FZ'))) AS surplus
FROM e_order A
LEFT JOIN e_requirement b
ON A.requirement_no = b.requirement_no
LEFT JOIN erp_project C
ON b.factory = C.project_no
WHERE 1 = 1
AND A.status IN ('WC')
AND (A.q_basic -
(SELECT COALESCE(SUM(e.q_basic), 0.00)
FROM e_order e
WHERE CAST(e.r_item AS INT) = A.ID
AND e.req_status NOT IN ('FZ'))) > 0.00
AND (C.project_name LIKE CONCAT('%', 'csg18098', '%') OR
C.project_no LIKE CONCAT('%', 'csg18098', '%'))
AND A.requirement_no LIKE CONCAT('%', '0000004390', '%');

修改后,关联的顺序就是表关联的顺序是((((a,e),b),c),d)。故d_1就是表示在where子句后的表e_order。

回看执行计划1,可以看到很多关键字(never executed)。其实在执行计划1中,(a,d_1)两个表关联后,返回的行数是0,所以之后加入连接的表其实并未执行实际连接操作,即b,c,d并未真的执行join操作。这个语句执行(a,d_1)两个表关联后就结束了。

而在where子句后删除对子查询结果大小判断后,表的连接顺序是(((a,b),c),d)。从执行计划2中可以看到,每个表都参与的join操作后,整个语句才执行结束。因此,时间比第一个执行计划的时间长了。

这里,子查询结果判断后返回的结果是0行。如果,不是0行呢?
我们把语句1中>0.00换成=0.000000,看看执行计划:

这个执行计划3,就比执行计划1和执行计划2都慢了。

pg执行计划分析小笔记的更多相关文章

  1. SQLServer查询执行计划分析 - 案例

    SQLServer查询执行计划分析 - 案例 http://pan.baidu.com/s/1pJ0gLjP 包括学习笔记.书.样例库

  2. MySQL学习系列2--MySQL执行计划分析EXPLAIN

    原文:MySQL学习系列2--MySQL执行计划分析EXPLAIN 1.Explain语法 EXPLAIN SELECT …… 变体:   EXPLAIN EXTENDED SELECT …… 将执行 ...

  3. MongoDB执行计划分析详解

    要保证数据库处于高效.稳定的状态,除了良好的硬件基础.高效高可用的数据库架构.贴合业务的数据模型之外,高效的查询语句也是不可少的.那么,如何查看并判断我们的执行计划呢?我们今天就来谈论下MongoDB ...

  4. MySQL学习系列2--MySQL执行计划分析EXPLAIN [原创]

    1.Explain语法 EXPLAIN SELECT …… 变体:   EXPLAIN EXTENDED SELECT …… 将执行计划“反编译”成SELECT语句,运行SHOW WARNINGS 可 ...

  5. MySQL执行计划分析

    原文:MySQL执行计划分析 一. 执行计划能告诉我们什么? SQL如何使用索引 联接查询的执行顺序 查询扫描的数据函数 二. 执行计划中的内容 SQL执行计划的输出可能为多行,每一行代表对一个数据库 ...

  6. MongoDB干货系列2-MongoDB执行计划分析详解(2)(转载)

    写在之前的话 作为近年最为火热的文档型数据库,MongoDB受到了越来越多人的关注,但是由于国内的MongoDB相关技术分享屈指可数,不少朋友向我抱怨无从下手. <MongoDB干货系列> ...

  7. oracle sql 执行计划分析

    转自http://itindex.net/detail/45962-oracle-sql-%E8%AE%A1%E5%88%92 一.首先创建表 SQL> show user USER is &q ...

  8. SQL执行计划分析

    explain执行计划中的字段以及含义在下面的博客中有详细讲述: https://blog.csdn.net/da_guo_li/article/details/79008016 执行计划能告诉我们什 ...

  9. mysql系列八、mysql数据库优化、慢查询优化、执行计划分析

    mysql的性能优化无法一蹴而就,必须一步一步慢慢来,从各个方面进行优化,最终性能就会有大的提升. 一.介绍 对mysql优化是一个综合性的技术,主要包括 表的设计合理化(符合3NF) 添加适当索引( ...

随机推荐

  1. css display block 和 inline

    根据CSS规范的规定,每一个网页元素都有一个display属性,用于确定该元素的类型,每一个元素都有默认的display属性值,比如div元素,它的默认display属性值为“block”,成为“块级 ...

  2. thrift简单示例 (基于C++)

    这个thrift的简单示例, 来源于官网 (http://thrift.apache.org/tutorial/cpp), 因为我觉得官网的例子已经很简单了, 所以没有写新的示例, 关于安装的教程, ...

  3. springdata jpa 关于分页@Query问题

    关于springdata jpa 分页问题相信很多小伙伴都遇到过,只要表中数量到达分页条件就会报错 废话少说直接上代码: @Query(nativeQuery = true, value = &quo ...

  4. node基础学习——path的处理与路径转换

    处理与转换路径path normalize该方法将非标准路径字符串转换为标准路径字符串,在转换过程中执行以下操作: ①解析路径字符串中的’..’字符串与’.’字符串,返回解析后的标准路径. ②将多个斜 ...

  5. 当电脑上有两个mysql时,如何让jmeter连接自己需要的那个mysql

    1.当有两个mysql时,修改其中一个的mysql端口号为3307 ,也可以是其他8788, 在文件my.ini中修改端口号为:3307:修改完之后记得重启mysql哦:dos下命令可以用net st ...

  6. Zookeeper windows环境安装

    环境要求:必须要有jdk环境,我自己是使用的 jdk1.8 1.安装jdk 2.安装Zookeeper. 在官网http://zookeeper.apache.org/下载zookeeper.我下载的 ...

  7. C++定义和初始化数组以及memset的使用(转)

    一.一维数组 静态 int array[100]; 定义了数组array,并未对数组进行初始化 静态 int array[100] = {1,2}: 定义并初始化了数组array 动态 int* ar ...

  8. [Codeforces 1251F]Red-White Fence

    Description 题库链接 给你 \(n\) 块白木板,\(k\) 块红木板,分别有各自的长度 \(h_i\).让你用这些木板组成一段围栏,要满足: 只用一块红木板,且所有白木板的长度均严格小于 ...

  9. 英语听力,如何成为更好的交谈着https://www.bilibili.com/video/av4279405?from=search&seid=5889429711390689339

    and how many of you know at least one person that you because you just do not want to talk to them.y ...

  10. MATLAB中运算符优先级

    下述运算符的优先级从低到高: 1.先决或(||): 2.先决与(&&): 3.逻辑或(|): 4.逻辑与(&): 5.等于类(<,<=,>,>=,==, ...