两种连接的表达 :left(right) join 和 (+)

简介:
稍微研究了一下 oracle 自己的join 和标准的join。主要表现在on ,where 关键字所起的作用不同,和连接本身的特性。
yang@ORACL> set autotrace on
yang@ORACL> select  *
  2  from a,b
  3  where a.id=b.id(+) and a.name like 'x%';

        ID NAME          ID NAME
---------- ----- ---------- -----
         1 x1             1 x1
         2 x2             2 x2
         4 x4
         3 x3
执行计划
---------------------------------------------------------------------------
| Id  | Operation          | Name | Rows  | Bytes | Cost (%CPU)| Time     |
---------------------------------------------------------------------------
|   0 | SELECT STATEMENT   |      |     4 |   280 |     7  (15)| 00:00:01 |
|*  1 |  HASH JOIN OUTER   |      |     4 |   280 |     7  (15)| 00:00:01 |
|*  2 |   TABLE ACCESS FULL| A    |     4 |   140 |     3   (0)| 00:00:01 |
|   3 |   TABLE ACCESS FULL| B    |     2 |    70 |     3   (0)| 00:00:01 |
---------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
   1 - access("A"."ID"="B"."ID"(+))
   2 - filter("A"."NAME" LIKE 'x%')
由执行计划可以看出先对a表进行过滤,让后进行与b表的left join,
--
yang@ORACL> select * from a left join b
  2  on a.id=b.id and a.name like 'x%';

        ID NAME          ID NAME
---------- ----- ---------- -----
         1 x1             1 x1
         2 x2             2 x2
         3 x3
         4 x4
         1 y1
         2 y2
         3 y3
         4 y4

已选择8行。

-----------------------------------------------------------------------------
| Id  | Operation            | Name | Rows  | Bytes | Cost (%CPU)| Time     |
-----------------------------------------------------------------------------
|   0 | SELECT STATEMENT     |      |     8 |   560 |    27   (0)| 00:00:01 |
|   1 |  NESTED LOOPS OUTER  |      |     8 |   560 |    27   (0)| 00:00:01 |
|   2 |   TABLE ACCESS FULL  | A    |     8 |   280 |     3   (0)| 00:00:01 |
|   3 |   VIEW               |      |     1 |    35 |     3   (0)| 00:00:01 |
|*  4 |    FILTER            |      |       |       |            |          |
|*  5 |     TABLE ACCESS FULL| B    |     1 |    35 |     3   (0)| 00:00:01 |
-----------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
   4 - filter("A"."NAME" LIKE 'x%')
   5 - filter("A"."ID"="B"."ID")
-- 先连接后筛选 (由 Table A 驱动 NESTED LOOPS OUTER ,每一次循环筛选记录
外连接中的on不需要过滤基表数据,过滤基表数据是在where里做的,on只是连接条件,根据连接条件找匹配的从表数据,找不到匹配的从表行,则置空。
yang@ORACL> select * from a left join b
  2  on a.id=b.id
  3  where a.name like 'x%';

        ID NAME          ID NAME
---------- ----- ---------- -----
         1 x1             1 x1
         2 x2             2 x2
         4 x4
         3 x3
执行计划
----------------------------------------------------------
Plan hash value: 1365417139

---------------------------------------------------------------------------
| Id  | Operation          | Name | Rows  | Bytes | Cost (%CPU)| Time     |
---------------------------------------------------------------------------
|   0 | SELECT STATEMENT   |      |     4 |   280 |     7  (15)| 00:00:01 |
|*  1 |  HASH JOIN OUTER   |      |     4 |   280 |     7  (15)| 00:00:01 |
|*  2 |   TABLE ACCESS FULL| A    |     4 |   140 |     3   (0)| 00:00:01 |
|   3 |   TABLE ACCESS FULL| B    |     2 |    70 |     3   (0)| 00:00:01 |
---------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
   1 - access("A"."ID"="B"."ID"(+))
   2 - filter("A"."NAME" LIKE 'x%')

Note
-----
   - dynamic sampling used for this statement

yang@ORACL> select * from a left join b
  2  on a.name like 'x%'
  3  where a.id=b.id;
        ID NAME          ID NAME
---------- ----- ---------- -----
         1 x1             1 x1
         2 x2             2 x2

执行计划
----------------------------------------------------------
Plan hash value: 652036164
---------------------------------------------------------------------------
| Id  | Operation          | Name | Rows  | Bytes | Cost (%CPU)| Time     |
---------------------------------------------------------------------------
|   0 | SELECT STATEMENT   |      |     1 |    70 |     7  (15)| 00:00:01 |
|*  1 |  HASH JOIN         |      |     1 |    70 |     7  (15)| 00:00:01 |
|*  2 |   TABLE ACCESS FULL| A    |     1 |    35 |     3   (0)| 00:00:01 |
|   3 |   TABLE ACCESS FULL| B    |     2 |    70 |     3   (0)| 00:00:01 |
---------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
   1 - access("A"."ID"="B"."ID")
   2 - filter("A"."NAME" LIKE 'x%')
需要说明的是:
 on关键字,的确是连接条件,它不能过滤基表。过滤不了的原因是on过滤掉的emp表结果,最后又被left outer join拿了回来。
 select e.ename,d.dname from emp e left outer join dept d on d.deptno=e.deptno and e.ename='SCOTT';
  伪代码
    for rec_e in ( select ename from emp e ) loop
        for rec_d in ( select dname from dept d where d.deptno = rec_e.deptno ) loop
             if rec_d != NULL and rec_e..ename='SCOTT' then
                show (rec_e.ename,rec_d.name);
             else
                show (rec_e.ename,null);   --on过滤掉的emp表结果,最后又被left outer join拿了回来。
             end if;
        end loop;
     end loop;
--伪代码部分内容参考了puber ccsnmoracle 的表述
相关文章
|
SQL Oracle 关系型数据库
解决:Oracle数据库中Left join on 后面为null时匹配不上
解决:Oracle数据库中Left join on 后面为null时匹配不上
264 0
|
SQL 关系型数据库 MySQL
MySql 别犯糊涂了! LEFT JOIN 的 ON 后接上筛选条件,多个条件会出事!
MySql 别犯糊涂了! LEFT JOIN 的 ON 后接上筛选条件,多个条件会出事!
2298 0
MySql 别犯糊涂了! LEFT JOIN 的 ON 后接上筛选条件,多个条件会出事!
|
18天前
|
关系型数据库 数据挖掘 数据库
解析数据库联结:应用与实践中的 INNER JOIN、LEFT JOIN、RIGHT JOIN、FULL OUTER JOIN 与 CROSS JOIN
解析数据库联结:应用与实践中的 INNER JOIN、LEFT JOIN、RIGHT JOIN、FULL OUTER JOIN 与 CROSS JOIN
37 2
|
18天前
|
关系型数据库 数据挖掘 数据库
解析数据库联结:应用与实践中的 INNER JOIN、LEFT JOIN、RIGHT JOIN、FULL OUTER JOIN 与 CROSS JOIN
解析数据库联结:应用与实践中的 INNER JOIN、LEFT JOIN、RIGHT JOIN、FULL OUTER JOIN 与 CROSS JOIN
41 1
|
关系型数据库 MySQL
left join、right join和join,傻傻分不清?
真的是一张图道清所有join的区别啊,可惜我还是看不懂,可能人比较懒,然后基本一个left join给我就是够用的了,所以就没怎么去仔细研究了,但是现实还是逼我去搞清楚,索性自己动手,总算理解图中的含义了,下面就听我一一道来。
102 1
|
SQL 关系型数据库 MySQL
Join,left join,right join(1)--连接原理(三十九)
Join,left join,right join(1)--连接原理(三十九)
|
SQL 语音技术 数据库
SQL基础【十五、join、Inner join、Left join、Right join、Full join】
SQL基础【十五、join、Inner join、Left join、Right join、Full join】
160 0
SQL基础【十五、join、Inner join、Left join、Right join、Full join】
|
SQL 数据库
left join 后用 on 还是 where,区别大了!
前天写SQL时本想通过 A left B join on and 后面的条件来使查出的两条记录变成一条,奈何发现还是有两条。 后来发现 join on and 不会过滤结果记录条数,只会根据and后的条件是否显示 B表的记录,A表的记录一定会显示。 不管and 后面的是A.id=1还是B.id=1,都显示出A表中所有的记录,并关联显示B中对应A表中id为1的记录或者B表中id为1的记录。
130 0
left join 后用 on 还是 where,区别大了!
|
SQL 关系型数据库 MySQL
正确理解Left join
通俗来讲,left join就是以左表作为主表,结果返回左表的所有记录,右表满足条件记录正常显示,满足条件记录使用NULL做填充,一般业务中我们需要显示左表全部记录时才会使用left join。另外,某些情况下MySQL优化器会将我们的left join改写为join,什么情况下MySQL会做这样的优化?
1372 0