join后的where,你清楚吗?


1.先看下示例表:ORACLE中,scott下的两张示例表

Sql代码 复制代码
  1. CREATE TABLE EMP   
  2. (   
  3.   EMPNO     NUMBER(4),   
  4.   ENAME     VARCHAR2(10 BYTE),   
  5.   JOB       VARCHAR2(9 BYTE),   
  6.   MGR       NUMBER(4),   
  7.   HIREDATE  DATE,   
  8.   SAL       NUMBER(7,2),   
  9.   COMM      NUMBER(7,2),   
  10.   DEPTNO    NUMBER(2)   
  11. )   
  12.   
  13. CREATE TABLE DEPT   
  14. (   
  15.   DEPTNO  NUMBER(2),   
  16.   DNAME   VARCHAR2(14 BYTE),   
  17.   LOC     VARCHAR2(13 BYTE)   
  18. )  
CREATE TABLE EMP
(
  EMPNO     NUMBER(4),
  ENAME     VARCHAR2(10 BYTE),
  JOB       VARCHAR2(9 BYTE),
  MGR       NUMBER(4),
  HIREDATE  DATE,
  SAL       NUMBER(7,2),
  COMM      NUMBER(7,2),
  DEPTNO    NUMBER(2)
)

CREATE TABLE DEPT
(
  DEPTNO  NUMBER(2),
  DNAME   VARCHAR2(14 BYTE),
  LOC     VARCHAR2(13 BYTE)
)

 

2.几条SQL引出的问题:(过滤条件在左表emp 中,sal != 3000)

Sql代码 复制代码
  1. --SQL1:全部当成join的条件   
  2. select e.* ,d.deptno , d.dname from emp e left join dept d    
  3. on e.deptno=d.deptno and sal != 3000  order by empno;   
  4.   
  5. --SQL2:先过滤再join   
  6. select e.*,d.deptno ,d.dname from (select * from emp  where sal != 3000 ) e left join dept d    
  7. on e.deptno=d.deptno order by empno;   
  8. --SQL3:先过滤再join   
  9. select e.* ,d.deptno , d.dname from emp e left join dept d    
  10. on e.deptno=d.deptno where sal != 3000 order by empno;  
--SQL1:全部当成join的条件
select e.* ,d.deptno , d.dname from emp e left join dept d 
on e.deptno=d.deptno and sal != 3000  order by empno;

--SQL2:先过滤再join
select e.*,d.deptno ,d.dname from (select * from emp  where sal != 3000 ) e left join dept d 
on e.deptno=d.deptno order by empno;
--SQL3:先过滤再join
select e.* ,d.deptno , d.dname from emp e left join dept d 
on e.deptno=d.deptno where sal != 3000 order by empno;

 SQL1生成的结果:



 

 SQL2和SQL3生成的结果一致:



 

SQL2或者3的执行计划是一致的:

Sql代码 复制代码
  1. 执行计划   
  2. ----------------------------------------------------------   
  3. Plan hash value: 1901738359   
  4.   
  5. ----------------------------------------------------------------------------   
  6. | Id  | Operation           | Name | Rows  | Bytes | Cost (%CPU)| Time     |   
  7. ----------------------------------------------------------------------------   
  8. |   0 | SELECT STATEMENT    |      |    12 |   600 |     8  (25)| 00:00:01 |   
  9. |   1 |  SORT ORDER BY      |      |    12 |   600 |     8  (25)| 00:00:01 |   
  10. |*  2 |   HASH JOIN OUTER   |      |    12 |   600 |     7  (15)| 00:00:01 |   
  11. |*  3 |    TABLE ACCESS FULL| EMP  |    12 |   444 |     3   (0)| 00:00:01 |   
  12. |   4 |    TABLE ACCESS FULL| DEPT |     4 |    52 |     3   (0)| 00:00:01 |   
  13. ----------------------------------------------------------------------------   
  14.   
  15. Predicate Information (identified by operation id):   
  16. ---------------------------------------------------   
  17.   
  18.    2 - access("EMP"."DEPTNO"="D"."DEPTNO"(+))   
  19.    3 - filter("SAL"<>3000)  
执行计划
----------------------------------------------------------
Plan hash value: 1901738359

----------------------------------------------------------------------------
| Id  | Operation           | Name | Rows  | Bytes | Cost (%CPU)| Time     |
----------------------------------------------------------------------------
|   0 | SELECT STATEMENT    |      |    12 |   600 |     8  (25)| 00:00:01 |
|   1 |  SORT ORDER BY      |      |    12 |   600 |     8  (25)| 00:00:01 |
|*  2 |   HASH JOIN OUTER   |      |    12 |   600 |     7  (15)| 00:00:01 |
|*  3 |    TABLE ACCESS FULL| EMP  |    12 |   444 |     3   (0)| 00:00:01 |
|   4 |    TABLE ACCESS FULL| DEPT |     4 |    52 |     3   (0)| 00:00:01 |
----------------------------------------------------------------------------

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

   2 - access("EMP"."DEPTNO"="D"."DEPTNO"(+))
   3 - filter("SAL"<>3000)

  可见是先过滤sal!=3000,再join

 

而SQL1会把整个e.deptno=d.deptno and sal != 3000 当成join条件,一旦满足sal=3000 就会找不到对应的dept.

  

所以凡遇见join后加的是左边表的过滤

Sql代码 复制代码
  1. select e.* ,d.deptno , d.dname from emp e left join dept d    
  2. on e.deptno=d.deptno where sal != 3000 order by empno;   
select e.* ,d.deptno , d.dname from emp e left join dept d 
on e.deptno=d.deptno where sal != 3000 order by empno; 

 

按照执行计划就可以理解成:

Sql代码 复制代码
  1. --先过滤再join   
  2. select e.*,d.deptno ,d.dname from (select * from emp  where sal != 3000 ) e left join dept d    
  3. on e.deptno=d.deptno order by empno;  
--先过滤再join
select e.*,d.deptno ,d.dname from (select * from emp  where sal != 3000 ) e left join dept d 
on e.deptno=d.deptno order by empno;

 所以,对于过滤的是左边表emp的条件sal!=3000,若是将过滤条件放在join条件之中,例如SQL1,会把它当成整个JOIN的条件,若是将过滤条件当成where放在join后,例如SQL3,就是先过滤再join,结果自然就不一样了。

 

3.还是几条SQL引出的问题:(过滤条件在右表 dept中,d.loc!='NEW YORK')

 

Sql代码 复制代码
  1. --SQL 4   
  2. select e.* ,  d.loc from emp e left join dept d on e.deptno=d.deptno and d.loc!='NEW YORK' order by empno;   
  3. --SQL 5   
  4. select e.*, d.loc from  emp e left join (select * from dept where dept.loc!='NEW YORK' ) d on e.deptno=d.deptno order by empno;   
  5. --SQL 6   
  6. select e.* , d.loc from emp e left join dept d on e.deptno=d.deptno where d.loc!='NEW YORK' order by empno;  
--SQL 4
select e.* ,  d.loc from emp e left join dept d on e.deptno=d.deptno and d.loc!='NEW YORK' order by empno;
--SQL 5
select e.*, d.loc from  emp e left join (select * from dept where dept.loc!='NEW YORK' ) d on e.deptno=d.deptno order by empno;
--SQL 6
select e.* , d.loc from emp e left join dept d on e.deptno=d.deptno where d.loc!='NEW YORK' order by empno;

 SQL4的执行计划:

Sql代码 复制代码
  1. 执行计划   
  2. ----------------------------------------------------------   
  3. Plan hash value: 1901738359   
  4.   
  5. ----------------------------------------------------------------------------   
  6. | Id  | Operation           | Name | Rows  | Bytes | Cost (%CPU)| Time     |   
  7. ----------------------------------------------------------------------------   
  8. |   0 | SELECT STATEMENT    |      |    14 |   672 |     8  (25)| 00:00:01 |   
  9. |   1 |  SORT ORDER BY      |      |    14 |   672 |     8  (25)| 00:00:01 |   
  10. |*  2 |   HASH JOIN OUTER   |      |    14 |   672 |     7  (15)| 00:00:01 |   
  11. |   3 |    TABLE ACCESS FULL| EMP  |    14 |   518 |     3   (0)| 00:00:01 |   
  12. |*  4 |    TABLE ACCESS FULL| DEPT |     3 |    33 |     3   (0)| 00:00:01 |   
  13. ----------------------------------------------------------------------------   
  14.   
  15. Predicate Information (identified by operation id):   
  16. ---------------------------------------------------   
  17.   
  18.    2 - access("E"."DEPTNO"="D"."DEPTNO"(+))   
  19.    4 - filter("D"."LOC"(+)<>'NEW YORK')  
执行计划
----------------------------------------------------------
Plan hash value: 1901738359

----------------------------------------------------------------------------
| Id  | Operation           | Name | Rows  | Bytes | Cost (%CPU)| Time     |
----------------------------------------------------------------------------
|   0 | SELECT STATEMENT    |      |    14 |   672 |     8  (25)| 00:00:01 |
|   1 |  SORT ORDER BY      |      |    14 |   672 |     8  (25)| 00:00:01 |
|*  2 |   HASH JOIN OUTER   |      |    14 |   672 |     7  (15)| 00:00:01 |
|   3 |    TABLE ACCESS FULL| EMP  |    14 |   518 |     3   (0)| 00:00:01 |
|*  4 |    TABLE ACCESS FULL| DEPT |     3 |    33 |     3   (0)| 00:00:01 |
----------------------------------------------------------------------------

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

   2 - access("E"."DEPTNO"="D"."DEPTNO"(+))
   4 - filter("D"."LOC"(+)<>'NEW YORK')

 

可见是先filter("D"."LOC"(+)<>'NEW YORK'),再join。相当于SQL5

所以SQL4和SQL5的执行结果是一样的。

 

但是SQL6的结果就很戏剧了,SQL6执行结果

 

 

 先join再过滤:

 select * from (select e.* ,d.loc from emp e left join dept d on e.deptno=d.deptno) where loc!='NEW YORK' order by empno

 

4.更经典的(过滤两张表共同的列):

Sql代码 复制代码
  1. select e.* ,d.deptno, d.dname from emp e left join dept d on e.deptno=d.deptno where d.deptno!=30 order by empno;  
select e.* ,d.deptno, d.dname from emp e left join dept d on e.deptno=d.deptno where d.deptno!=30 order by empno;

 

Sql代码 复制代码
  1. 执行计划   
  2. ----------------------------------------------------------   
  3. Plan hash value: 3357797783   
  4.   
  5. ----------------------------------------------------------------------------   
  6. | Id  | Operation           | Name | Rows  | Bytes | Cost (%CPU)| Time     |   
  7. ----------------------------------------------------------------------------   
  8. |   0 | SELECT STATEMENT    |      |     9 |   450 |     8  (25)| 00:00:01 |   
  9. |   1 |  SORT ORDER BY      |      |     9 |   450 |     8  (25)| 00:00:01 |   
  10. |*  2 |   HASH JOIN         |      |     9 |   450 |     7  (15)| 00:00:01 |   
  11. |*  3 |    TABLE ACCESS FULL| DEPT |     3 |    39 |     3   (0)| 00:00:01 |   
  12. |*  4 |    TABLE ACCESS FULL| EMP  |     9 |   333 |     3   (0)| 00:00:01 |   
  13. ----------------------------------------------------------------------------   
  14.   
  15. Predicate Information (identified by operation id):   
  16. ---------------------------------------------------   
  17.   
  18.    2 - access("E"."DEPTNO"="D"."DEPTNO")   
  19.    3 - filter("D"."DEPTNO"<>30)   
  20.    4 - filter("E"."DEPTNO"<>30)  
执行计划
----------------------------------------------------------
Plan hash value: 3357797783

----------------------------------------------------------------------------
| Id  | Operation           | Name | Rows  | Bytes | Cost (%CPU)| Time     |
----------------------------------------------------------------------------
|   0 | SELECT STATEMENT    |      |     9 |   450 |     8  (25)| 00:00:01 |
|   1 |  SORT ORDER BY      |      |     9 |   450 |     8  (25)| 00:00:01 |
|*  2 |   HASH JOIN         |      |     9 |   450 |     7  (15)| 00:00:01 |
|*  3 |    TABLE ACCESS FULL| DEPT |     3 |    39 |     3   (0)| 00:00:01 |
|*  4 |    TABLE ACCESS FULL| EMP  |     9 |   333 |     3   (0)| 00:00:01 |
----------------------------------------------------------------------------

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

   2 - access("E"."DEPTNO"="D"."DEPTNO")
   3 - filter("D"."DEPTNO"<>30)
   4 - filter("E"."DEPTNO"<>30)

 

此时2张表都被过滤后( 3 - filter("D"."DEPTNO"<>30)  4 - filter("E"."DEPTNO"<>30))再JOIN。

 

查询结果:

 

 

呵呵,小心你的where和join吧,先想清楚业务需求,再想清楚SQL原理,才能查出正确的结果,否则就悲剧了。

附件scott.rar是导出的emp和dept的备份。oracle导出版本是 Release 10.2.0.1.0,低于这个版本的,将无法导入我导出的这个文件。

 

附SQL:

Sql代码 复制代码
  1. --全部当成join的条件   
  2. select e.* ,d.deptno , d.dname from emp e left join dept d    
  3. on e.deptno=d.deptno and sal != 3000  order by empno;   
  4.   
  5. --先过滤再join   
  6. select e.*,d.deptno ,d.dname from (select * from emp  where sal != 3000 ) e left join dept d    
  7. on e.deptno=d.deptno order by empno;   
  8. --先过滤再join   
  9. select e.* ,d.deptno , d.dname from emp e left join dept d    
  10. on e.deptno=d.deptno where sal != 3000 order by empno;   
  11.   
  12.   
  13. --SQL 4   
  14. select e.* ,  d.loc from emp e left join dept d on e.deptno=d.deptno and d.loc!='NEW YORK' order by empno;   
  15. --SQL 5   
  16. select e.*, d.loc from  emp e left join (select * from dept where dept.loc!='NEW YORK' ) d on e.deptno=d.deptno order by empno;   
  17. --SQL 6   
  18. select e.* ,d.loc from emp e left join dept d on e.deptno=d.deptno where d.loc!='NEW YORK' order by empno;   
  19. select * from (select e.* ,d.loc from emp e left join dept d on e.deptno=d.deptno) where loc!='NEW YORK' order by empno   
  20.   
  21.   
  22. select e.* , d.deptno,d.dname from emp e left join dept d on e.deptno=d.deptno and d.deptno!=30 order by empno;   
  23.   
  24. --先过滤再join   
  25. select e.*,d.deptno,d.dname from  emp e left join (select * from dept where dept.deptno!=30 ) d on e.deptno=d.deptno order by empno;   
  26.   
  27. select e.* ,d.deptno, d.dname from emp e left join dept d on e.deptno=d.deptno where d.deptno!=30 order by empno;  
--全部当成join的条件
select e.* ,d.deptno , d.dname from emp e left join dept d 
on e.deptno=d.deptno and sal != 3000  order by empno;

--先过滤再join
select e.*,d.deptno ,d.dname from (select * from emp  where sal != 3000 ) e left join dept d 
on e.deptno=d.deptno order by empno;
--先过滤再join
select e.* ,d.deptno , d.dname from emp e left join dept d 
on e.deptno=d.deptno where sal != 3000 order by empno;


--SQL 4
select e.* ,  d.loc from emp e left join dept d on e.deptno=d.deptno and d.loc!='NEW YORK' order by empno;
--SQL 5
select e.*, d.loc from  emp e left join (select * from dept where dept.loc!='NEW YORK' ) d on e.deptno=d.deptno order by empno;
--SQL 6
select e.* ,d.loc from emp e left join dept d on e.deptno=d.deptno where d.loc!='NEW YORK' order by empno;
select * from (select e.* ,d.loc from emp e left join dept d on e.deptno=d.deptno) where loc!='NEW YORK' order by empno


select e.* , d.deptno,d.dname from emp e left join dept d on e.deptno=d.deptno and d.deptno!=30 order by empno;

--先过滤再join
select e.*,d.deptno,d.dname from  emp e left join (select * from dept where dept.deptno!=30 ) d on e.deptno=d.deptno order by empno;

select e.* ,d.deptno, d.dname from emp e left join dept d on e.deptno=d.deptno where d.deptno!=30 order by empno;

 

 

 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值