标量子查询的语句:

select /*+ GATHER_PLAN_STATISTICS  dwtest */ empno,
(select count(*) from DEPT1 b where b.id = a.id) as d,
(select sum(x) from DEPT1 b where b.id = a.id) as e
from EMP1 a --where a.id in (1,2,3,4,5,6,7,8,9,11,12,13) 

创建表:

create table DEPT1 as select  * from DEPT;
insert into DEPT1 select * from DEPT1; ---多执行几次 create table emp1 as select * from emp;
insert into emp1 select * from emp1; ---多执行几次 alter table DEPT1 add id number;
update DEPT1 set id=rownum; alter table emp1 add id number;
update emp1 set id=rownum;
create index idx_emp1 on emp1(id);

create index idx_DEPT1 on dept1(id);

改写后的语句:

select /*+ GATHER_PLAN_STATISTICS  dwtes2 */ empno,d,e from EMP1 a
left join (select count(1)d,id,sum(x) e from DEPT1 group by id) b on a.id=b.id --where a.id in (1,2,3,4,5,6,7,8,9,11,12,13);

  在放开where条件时,标量子查询时的多次索引范围扫描,导致cost较高,性能比左连接方式要差些,从下图看的不是很明显,如果从monitor看耗时更直观些。得出结论是:左连接改写后效率比标量子查询稍好,但是没有明显提升。

----
SQL_ID 8mgcw7x9x16rq, child number 0
-------------------------------------
select /*+ GATHER_PLAN_STATISTICS dwtest */ empno, (select
count(*) from DEPT1 b where b.id = a.id) as d, (select sum(x)
from DEPT1 b where b.id = a.id) as e from EMP1 a where a.id in
(1,2,3,4,5,6,7,8,9,11,12,13) Plan hash value: 3470857716 -------------------------------------------------------------------------------------------------------------
| Id | Operation | Name | Starts | E-Rows | A-Rows | A-Time | Buffers | Reads |
-------------------------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | | 12 |00:00:00.01 | 9 | 4 |
| 1 | SORT AGGREGATE | | 12 | 1 | 12 |00:00:00.01 | 8 | 1 |
|* 2 | INDEX RANGE SCAN | IDX_DEPT1 | 12 | 1 | 12 |00:00:00.01 | 8 | 1 |
| 3 | SORT AGGREGATE | | 12 | 1 | 12 |00:00:00.01 | 10 | 0 |
| 4 | TABLE ACCESS BY INDEX ROWID| DEPT1 | 12 | 1 | 12 |00:00:00.01 | 10 | 0 |
|* 5 | INDEX RANGE SCAN | IDX_DEPT1 | 12 | 1 | 12 |00:00:00.01 | 8 | 0 |
| 6 | INLIST ITERATOR | | 1 | | 12 |00:00:00.01 | 9 | 4 |
| 7 | TABLE ACCESS BY INDEX ROWID| EMP1 | 12 | 12 | 12 |00:00:00.01 | 9 | 4 |
|* 8 | INDEX RANGE SCAN | IDX_EMP1 | 12 | 12 | 12 |00:00:00.01 | 8 | 4 |
------------------------------------------------------------------------------------------------------------- Predicate Information (identified by operation id):
--------------------------------------------------- 2 - access("B"."ID"=:B1)
5 - access("B"."ID"=:B1)
8 - access(("A"."ID"=1 OR "A"."ID"=2 OR "A"."ID"=3 OR "A"."ID"=4 OR "A"."ID"=5 OR "A"."ID"=6 OR
"A"."ID"=7 OR "A"."ID"=8 OR "A"."ID"=9 OR "A"."ID"=11 OR "A"."ID"=12 OR "A"."ID"=13)) -----
SQL_ID dpwyqsf1rch2g, child number 0
-------------------------------------
select /*+ GATHER_PLAN_STATISTICS dwtes2 */ empno,d,e from EMP1 a left
join (select count(1)d,id,sum(x) e from DEPT1 group by id) b on
a.id=b.id where a.id in (1,2,3,4,5,6,7,8,9,11,12,13) Plan hash value: 1193336691 ----------------------------------------------------------------------------------------------------------------------------------
| Id | Operation | Name | Starts | E-Rows | A-Rows | A-Time | Buffers | OMem | 1Mem | Used-Mem |
----------------------------------------------------------------------------------------------------------------------------------
| 0 | SELECT STATEMENT | | 1 | | 12 |00:00:00.01 | 19 | | | |
|* 1 | HASH JOIN OUTER | | 1 | 12 | 12 |00:00:00.01 | 19 | 1969K| 1969K| 1407K (0)|
| 2 | INLIST ITERATOR | | 1 | | 12 |00:00:00.01 | 9 | | | |
| 3 | TABLE ACCESS BY INDEX ROWID | EMP1 | 12 | 12 | 12 |00:00:00.01 | 9 | | | |
|* 4 | INDEX RANGE SCAN | IDX_EMP1 | 12 | 12 | 12 |00:00:00.01 | 8 | | | |
| 5 | VIEW | | 1 | 12 | 12 |00:00:00.01 | 10 | | | |
| 6 | HASH GROUP BY | | 1 | 12 | 12 |00:00:00.01 | 10 | 1116K| 1116K| 2222K (0)|
| 7 | INLIST ITERATOR | | 1 | | 12 |00:00:00.01 | 10 | | | |
| 8 | TABLE ACCESS BY INDEX ROWID| DEPT1 | 12 | 12 | 12 |00:00:00.01 | 10 | | | |
|* 9 | INDEX RANGE SCAN | IDX_DEPT1 | 12 | 12 | 12 |00:00:00.01 | 8 | | | |
---------------------------------------------------------------------------------------------------------------------------------- Predicate Information (identified by operation id):
--------------------------------------------------- 1 - access("A"."ID"="B"."ID")
4 - access(("A"."ID"=1 OR "A"."ID"=2 OR "A"."ID"=3 OR "A"."ID"=4 OR "A"."ID"=5 OR "A"."ID"=6 OR "A"."ID"=7 OR
"A"."ID"=8 OR "A"."ID"=9 OR "A"."ID"=11 OR "A"."ID"=12 OR "A"."ID"=13))
9 - access(("ID"=1 OR "ID"=2 OR "ID"=3 OR "ID"=4 OR "ID"=5 OR "ID"=6 OR "ID"=7 OR "ID"=8 OR "ID"=9 OR "ID"=11 OR
"ID"=12 OR "ID"=13))

  在没有where条件时,上例的索引选择性非常好,标量子查询时的多次索引范围扫描,与左连接方式的对两表的全表扫描的hash排序cost差不多,导致两种方式效率相差无几。得出结论是:在关联索引选择性非常好时,左连接改写后效率比标量子查询差不多,但是如果索引选择性一般时,左连接效果要好。

  但是在关联条件没有索引时,emp1表多少行,就要对dept是乘2次的全表扫描,此时就会导致cost非常高。

所以,尽量使用左连接加分组来优化

标量子查询加聚合函数sql改写一的更多相关文章

  1. 【2017-03-12】SQL Sever 子查询、聚合函数

    一.子查询 子查询:把一条查询语句,当做值来使用子句的查询结果必须是一列子句可以返回多行数据,但必须是一列 子句返回的值为一个值的时候: 例如: 我只知道c026这个编号,我要查询比这个车价格低的全部 ...

  2. 标量子查询SQL改写

    一网友说下面sql跑的好慢,让我看看 sql代码: select er, cid, pid, tbl, zs, sy, (select count(sr.mobile_tele_no) from tb ...

  3. 18 12 06 sql 的 基本语句 查询 条件查询 逻辑运算符 模糊查询 范围查询 排序 聚合函数 分组 分页 连接查询 自关联 子查询

    -- 数据的准备 -- 创建一个数据库 create database python_test charset=utf8; -- 使用一个数据库 use python_test; -- 显示使用的当前 ...

  4. 优化有标量子查询的SQL

    数据库环境:SQL SERVER 2008R2 今天在数据库中抓出一条比较耗费资源的SQL,只返回904条数据,居然跑了40多分钟.SQL及对应的数据量如下图: SELECT saft04.cur_y ...

  5. SQL Server的优化器会缓存标量子查询结果集吗

    在这篇博客"ORACLE当中自定义函数性优化浅析"中,我们介绍了通过标量子查询缓存来优化函数性能: 标量子查询缓存(scalar subquery caching)会通过缓存结果减 ...

  6. [20180626]函数与标量子查询14.txt

    [20180626]函数与标量子查询14.txt --//前面看http://www.cnblogs.com/kerrycode/p/9099507.html链接,里面提到: 通俗来将,当使用标量子查 ...

  7. 标量子查询调优SQL

    fxnjbmhkk4pp4 select /*+ leading (wb,sb,qw) */ 'blocker('||wb.holding_session||':'||sb.username||')- ...

  8. Oracle sql优化之分析函数优化标量子查询

    待优化语句如下 select a.code as code, a.m_code as m_code,a.stktype as f_stype,a.e_year as e_year, b.sname a ...

  9. SQL优化-标量子查询(数据仓库设计的隐患-标量子查询)

    项目数据库集群出现了大规模节点宕机问题.经查询,问题在于几张表被锁.主要问题在于近期得几个项目在数据库SQL编写时大量使用了标量子查询. 为确定为题确实是由于数据表访问量超过单节点限制,做了一些测试. ...

  10. 【SQL基础】【记住重命名】高级查询:聚合函数(四舍五入)、分组过滤、排序、

    〇.概述 1.功能概述 高级查询:聚合函数(四舍五入).分组过滤.排序. 2.建表语句 drop table if exists user_profile; CREATE TABLE `user_pr ...

随机推荐

  1. js循环判断创建新对象放数组中

    原效果 之后效果: <!doctype html> <html lang="en"> <head> <meta charset=" ...

  2. 轻量级CI/CD发布部署环境搭建及使用_03_docker安装harbor

    轻量级CI/CD发布部署环境搭建及使用_03_docker安装harbor 授人以鱼不如授人以渔,如果说的别人都没明白,说明自己实际也不是太明白   1,下载docker-compose sudo c ...

  3. 如何在 JavaScript 中使用媒体查询

    前言 说起媒体查询想必大家最先想到的都是CSS中@media,没错,这是我们最常用的媒体查询方法,主要用来为我们的网站做适配处理. 比如: h1 { font-size: 2rem; color: g ...

  4. ImGui引入深度测试的方法

    You can use AddCallback() on a given ImDrawList:: to register functions to be run during rendering a ...

  5. [C#]判断一个IP是否在某个IP段内

    关于IP地址 IPv4地址是由4段0-255的数字组成的,例如:a.b.c.d(0≤a,b,c,d≤255),IPv4也叫32位地址,为什么是32位呢,因为把每一段转换成二进制后,它的取值范围就是00 ...

  6. freeRTOS移植成功

    今天来学习如何移植freeRTOS 也算是走了很多的坑,总算是把系统跑起来了 相关的教程网上也有比较详细的,本文主要说说自己踩的坑 一些汇编文件报错的问题 这个问题的原因是因为网上大部分的移植说明都是 ...

  7. 测开-面试题-Java基础

    1 垃圾回收机制? 答: 一.手动垃圾回收机制(C/C++)手动:使用过的对象必须要程序员自己来回收 缺点: 1.若程序员忘记及时回收--对象会一直在内存中,若程序运行时间很长,内存中存在大量垃圾,空 ...

  8. offer_48

    题目:写一个函数,求两个整数之和,要求在函数体内不得使用+.-.*./四则运算符号. 思路: /* 首先看十进制是如何做的: 5+7=12,三步走 第一步:相加各位的值,不算进位,得到2. 第二步:计 ...

  9. 1487. 保证文件名唯一 (Medium)

    问题描述 1487. 保证文件名唯一 (Medium) 给你一个长度为 n 的字符串数组 names .你将会在文件系统中创建 n 个文件夹:在第 i 分钟,新建名为 names[i] 的文件夹. 由 ...

  10. 实验五Elasticsearch+Kibana展示爬虫数据

    安装elasticsearch-rtf Elasticsearch-rtf相比于elasticsearch而言多加了一些插件,因此我们选择安装Elasticsearch-rtf是一个不错的选择.在安装 ...