[20180808]exists and not exists.txt

--//生产系统遇到的一个性能问题,通过例子来说明:

1.环境:
SCOTT@test01p> @ ver1

PORT_STRING                    VERSION        BANNER                                                                               CON_ID
------------------------------ -------------- -------------------------------------------------------------------------------- ----------
IBMPC/WIN_NT64-9.1.0           12.1.0.1.0     Oracle Database 12c Enterprise Edition Release 12.1.0.1.0 - 64bit Production              0

SCOTT@test01p> create table t1 as select * from all_objects;
Table created.

SCOTT@test01p> create table t2 as select object_id,'1' flag from t1;
Table created.

SCOTT@test01p> select max(object_id) from t2;
MAX(OBJECT_ID)
--------------
        107828

SCOTT@test01p> update t2 set flag='0' where object_id=107828;
1 row updated.

SCOTT@test01p> commit ;
Commit complete.

SCOTT@test01p> create index i_t2_flag on t2(flag);
Index created.

--//分析表,并且t2的flag字段建立直方图.
execute sys.dbms_stats.gather_table_stats ( OwnName => user,TabName => 't1',Estimate_Percent => NULL,Method_Opt => 'FOR ALL COLUMNS SIZE 1 ',Cascade => True ,No_Invalidate => false);
execute sys.dbms_stats.gather_table_stats ( OwnName => user,TabName => 't2',Estimate_Percent => NULL,Method_Opt => 'FOR ALL COLUMNS SIZE 1 for columns flag size 10  ',Cascade => True ,No_Invalidate => false);
`
2.测试:
SCOTT@test01p> alter session set statistics_level=all;
Session altered.

SCOTT@test01p> select object_name  from t1 where not exists (select 1 from t2 where t2.object_id=t1.object_id and t2.flag='1' );
OBJECT_NAME
--------------------
T1

SCOTT@test01p> @ dpc '' ''
PLAN_TABLE_OUTPUT
-------------------------------------
SQL_ID  d4qcxhmwy49r1, child number 0
-------------------------------------
select object_name  from t1 where not exists (select 1 from t2 where
t2.object_id=t1.object_id and t2.flag='1' )

Plan hash value: 629543484

-------------------------------------------------------------------------------------------------------------------------------------------------------------------
| Id  | Operation            | Name | Starts | E-Rows |E-Bytes|E-Temp | Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers | Reads  |  OMem |  1Mem | Used-Mem |
-------------------------------------------------------------------------------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT     |      |      1 |        |       |       |   728 (100)|          |      1 |00:00:00.24 |    1667 |   1511 |       |       |          |
|*  1 |  HASH JOIN RIGHT ANTI|      |      1 |    899 | 33263 |  1672K|   728   (1)| 00:00:01 |      1 |00:00:00.24 |    1667 |   1511 |  5536K|  3056K| 5658K (0)|
|*  2 |   TABLE ACCESS FULL  | T2   |      1 |  89876 |   614K|       |    46   (3)| 00:00:01 |  89876 |00:00:00.02 |     152 |      0 |       |       |          |
|   3 |   TABLE ACCESS FULL  | T1   |      1 |  89877 |  2633K|       |   421   (1)| 00:00:01 |  89877 |00:00:00.11 |    1515 |   1511 |       |       |          |
-------------------------------------------------------------------------------------------------------------------------------------------------------------------

Query Block Name / Object Alias (identified by operation id):
-------------------------------------------------------------

1 - SEL$5DA710D3
   2 - SEL$5DA710D3 / T2@SEL$2
   3 - SEL$5DA710D3 / T1@SEL$1

Predicate Information (identified by operation id):
---------------------------------------------------

1 - access("T2"."OBJECT_ID"="T1"."OBJECT_ID")
   2 - filter("T2"."FLAG"='1')

--//仔细看id-2.过滤条件是   2 - filter("T2"."FLAG"='1').这样即使你建立索引在t2.flag也不会使用.因为flag='1'占大多数.
--//实际上对于当前应用改成如下是等效的.因为flag仅仅两种取值'0','1'.

SCOTT@test01p> select object_name  from t1 where  exists (select 1 from t2 where t2.object_id=t1.object_id and t2.flag='0' );
OBJECT_NAME
--------------------
T1

SCOTT@test01p> @ dpc '' ''
PLAN_TABLE_OUTPUT
-------------------------------------
SQL_ID  1y5xvtwz0u11f, child number 0
-------------------------------------
select object_name  from t1 where  exists (select 1 from t2 where
t2.object_id=t1.object_id and t2.flag='0' )
Plan hash value: 1273788863
--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
| Id  | Operation                            | Name      | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers | Reads  |  OMem |  1Mem | Used-Mem |
--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT                     |           |      1 |        |       |   423 (100)|          |      1 |00:00:00.19 |    1518 |   1512 |       |       |          |
|*  1 |  HASH JOIN RIGHT SEMI                |           |      1 |      1 |    37 |   423   (1)| 00:00:01 |      1 |00:00:00.19 |    1518 |   1512 |  2168K|  2168K|  697K (0)|
|   2 |   TABLE ACCESS BY INDEX ROWID BATCHED| T2        |      1 |      1 |     7 |     2   (0)| 00:00:01 |      1 |00:00:00.04 |       3 |      1 |       |       |          |
|*  3 |    INDEX RANGE SCAN                  | I_T2_FLAG |      1 |      1 |       |     1   (0)| 00:00:01 |      1 |00:00:00.04 |       2 |      1 |       |       |          |
|   4 |   TABLE ACCESS FULL                  | T1        |      1 |  89877 |  2633K|   421   (1)| 00:00:01 |  89877 |00:00:00.12 |    1515 |   1511 |       |       |          |
--------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
Query Block Name / Object Alias (identified by operation id):
-------------------------------------------------------------
   1 - SEL$5DA710D3
   2 - SEL$5DA710D3 / T2@SEL$2
   3 - SEL$5DA710D3 / T2@SEL$2
   4 - SEL$5DA710D3 / T1@SEL$1
Predicate Information (identified by operation id):
---------------------------------------------------
   1 - access("T2"."OBJECT_ID"="T1"."OBJECT_ID")
   3 - access("T2"."FLAG"='0')

--//实际上到具体应用object_id字段是主键,如果在上面建立索引,逻辑读更小.
CREATE UNIQUE INDEX SCOTT.pk_t1 ON SCOTT.T1 (OBJECT_ID);
ALTER TABLE SCOTT.T1 ADD CONSTRAINT pk_t1  PRIMARY KEY (OBJECT_ID);

CREATE UNIQUE INDEX SCOTT.pk_t2 ON SCOTT.T2 (OBJECT_ID);
ALTER TABLE SCOTT.T2 ADD CONSTRAINT pk_t2  PRIMARY KEY (OBJECT_ID);

SCOTT@test01p> select object_name  from t1 where  exists (select 1 from t2 where t2.object_id=t1.object_id and t2.flag='0' );
OBJECT_NAME
--------------------
T1

SCOTT@test01p> @ dpc '' ''
PLAN_TABLE_OUTPUT
-------------------------------------
SQL_ID  1y5xvtwz0u11f, child number 0
-------------------------------------
select object_name  from t1 where  exists (select 1 from t2 where
t2.object_id=t1.object_id and t2.flag='0' )
Plan hash value: 4193600567
----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
| Id  | Operation                              | Name      | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers | Reads  |  OMem |  1Mem | Used-Mem |
----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
|   0 | SELECT STATEMENT                       |           |      1 |        |       |     3 (100)|          |      1 |00:00:00.04 |       6 |      2 |       |       |          |
|   1 |  NESTED LOOPS                          |           |      1 |        |       |            |          |      1 |00:00:00.04 |       6 |      2 |       |       |          |
|   2 |   NESTED LOOPS                         |           |      1 |      1 |    37 |     3   (0)| 00:00:01 |      1 |00:00:00.03 |       5 |      1 |       |       |          |
|   3 |    SORT UNIQUE                         |           |      1 |      1 |     7 |     2   (0)| 00:00:01 |      1 |00:00:00.01 |       3 |      0 |  2048 |  2048 | 2048  (0)|
|   4 |     TABLE ACCESS BY INDEX ROWID BATCHED| T2        |      1 |      1 |     7 |     2   (0)| 00:00:01 |      1 |00:00:00.01 |       3 |      0 |       |       |          |
|*  5 |      INDEX RANGE SCAN                  | I_T2_FLAG |      1 |      1 |       |     1   (0)| 00:00:01 |      1 |00:00:00.01 |       2 |      0 |       |       |          |
|*  6 |    INDEX UNIQUE SCAN                   | PK_T1     |      1 |      1 |       |     0   (0)|          |      1 |00:00:00.03 |       2 |      1 |       |       |          |
|   7 |   TABLE ACCESS BY INDEX ROWID          | T1        |      1 |      1 |    30 |     1   (0)| 00:00:01 |      1 |00:00:00.01 |       1 |      1 |       |       |          |
----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------
Query Block Name / Object Alias (identified by operation id):
-------------------------------------------------------------
   4 - SEL$5DA710D3 / T2@SEL$2
   5 - SEL$5DA710D3 / T2@SEL$2
   6 - SEL$5DA710D3 / T1@SEL$1
   7 - SEL$5DA710D3 / T1@SEL$1
Predicate Information (identified by operation id):
---------------------------------------------------
   5 - access("T2"."FLAG"='0')
   6 - access("T2"."OBJECT_ID"="T1"."OBJECT_ID")
Note
-----
   - this is an adaptive plan

--//而select object_name  from t1 where not exists (select 1 from t2 where t2.object_id=t1.object_id and t2.flag='1' );执行计划不变.不再贴出.
--//我有时候想开发写sql代码过脑子没有,有时候真的很无语很无奈...

[20180808]exists and not exists.txt的更多相关文章

  1. 转【】浅谈sql中的in与not in,exists与not exists的区别_

    浅谈sql中的in与not in,exists与not exists的区别   1.in和exists in是把外表和内表作hash连接,而exists是对外表作loop循环,每次loop循环再对内表 ...

  2. 浅谈sql中的in与not in,exists与not exists的区别

    转 浅谈sql中的in与not in,exists与not exists的区别   12月12日北京OSC源创会 —— 开源技术的年终盛典 »   sql exists in 1.in和exists ...

  3. MySQL 子查询 EXISTS 和 NOT EXISTS(转)

    MySQL EXISTS 和 NOT EXISTS 子查询 MySQL EXISTS 和 NOT EXISTS 子查询语法如下: SELECT ... FROM table WHERE EXISTS ...

  4. MySQL 子查询 EXISTS 和 NOT EXISTS

    MySQL EXISTS 和 NOT EXISTS 子查询 MySQL EXISTS 和 NOT EXISTS 子查询语法如下: SELECT ... FROM table WHERE EXISTS ...

  5. oracle中的exists 和not exists 用法 in与exists语句的效率问题

    博文来源(oracle中的exists 和not exists 用法):http://chenshuai365-163-com.iteye.com/blog/1003247 博文来源(  in与exi ...

  6. (转)sql中 in 、not in 、exists、not exists 用法和差别

    exists (sql 返回结果集为真)  not exists (sql 不返回结果集为真)  如下:  表A  ID NAME  1    A1  2    A2  3  A3 表B  ID AI ...

  7. Mysql数据库中的EXISTS和NOT EXISTS

    SQL语言中没有蕴含逻辑运算.但是,可以利用谓词演算将一个逻辑蕴含的谓词等价转换为:p->q ≡┐p∨q. 我们通过一个具体的题目来分析:(具体的表和数据详见文章:Mysql数据库中的EXIST ...

  8. sql中 in 、not in 、exists、not exists 使用方法和区别

    % 的一类. NOT IN:通过 NOT IN keyword引入的子查询也返回一列零值或很多其它值. 以下查询查找没有出版过商业书籍的出版商的名称. SELECT pub_name FROM pub ...

  9. C#中当程序的访问权限不足时,Directory.Exists和File.Exists方法不会抛出异常报错

    有些时候,我们开发的C#应用程序的执行账号,可能没有对一些文件夹和文件的访问权限,当我们使用Directory.Exists和File.Exists方法去判断这些文件夹和文件是否存在的时候,Direc ...

随机推荐

  1. 《JavaScript总结》js模块化

    模块化开发,可以让代码易于扩展.便于日后维护. ES6中的模块化 我们先了解一下 export(导出) 和 import(导入) 这两个关键字. 新建一个文件a.js 并且导出变量test expor ...

  2. Mybatis解析mapper

    众所周知,接口是不能被实例化的,但是日常开发中,我们经常能直接使用dao层对象的方法,这又是为什么呢. 带着这些问题,我们看下mybatis内部做了那些操作. Mapper解析 上文我们描述了myba ...

  3. 项目ITP(一) 二维码

    前言 系列文章:[传送门] 上几周碌碌无为,不行啊不行啊.博客园,不知道你几时改版.老家了,我不会忘记你呢.呵呵,我也会在os,csdn更新的.每天一搏,不管有用没用. 正文 正文先有项目起步,项目中 ...

  4. java正则表达式的忽略大小写

    (?i)abc 表示abc都忽略大小写  a(?i)bc 表示bc忽略大小写  a((?i)b)c 表示只有b忽略大小写

  5. MFC控件编程之复选框单选框分组框

    MFC控件编程之复选框单选框分组框 一丶分组框 分组框 英文叫做 GroubBox 添加了分组框主要就是分组.好看.不重点介绍 二丶单选框 英文: Raido Button 单选框需要注意的事项 1. ...

  6. 【原创】sizeof运算符总结

    sizeof运算符返回一条表达式或一个类型名字的所占字节数,返回值为size_t的常量表达式,注意:sizeof右结合,且为编译时计算,而非运行时 两种形式:sizeof (type)和sizeof ...

  7. Nacos发布0.5.0版本,轻松玩转动态 DNS 服务

    阿里巴巴微服务开源项目Nacos于近期发布v0.5.0版本,该版本主要包括了DNS-basedService Discovery,对Java 11的支持,持续优化Nacos产品用户体验,更深度的与Sp ...

  8. DRDS SQL 审计与分析——全面洞察 SQL 之利器

    背景 数据库存储着系统的核心数据,其安全方面的问题在传统环境中已经成为泄漏和被篡改的重要根源.而在云端,数据库所面临的威胁被进一步的放大.因此,对云数据库的操作行为尤其是全量 SQL 执行记录的审计日 ...

  9. kubernetes的安装

    获取源码 最新安装包下载地址,GitHub下载地址 本次实验的1.10.0的二进制包下载,百度网盘 机器环境 Kubernetes Roles IP地址 Hostname Master 192.168 ...

  10. Nacos系列:基于Nacos的配置中心

    前言 在看正文之前,我想请你回顾一下自己待过的公司都是怎么管理配置的,我想应该会有以下几种方式: 1.硬编码 没有什么配置不配置的,直接写在代码里面,比如使用常量类 优势:对开发友好,开发清楚地知道代 ...