`

Oracle中left join中右表的限制条件

阅读更多

无过滤条件的LEFT JOIN

SQL中最简单形式的LEFT JOIN,是直接根据关联字段,以左表为基准,对右表进行匹配。在SELECT语句中选取的字段,如果有右表的记录(一般都是需要右表的某些记录的),取出配对成功的右表记录中对应的这个字段的值;否则,直接置NULL。这本身就是LEFT JOIN的特点:保证左表记录完整,右表只是辅助匹配。

直接看例子,为了演示,准备了两张测试表test1,test2:

 

select * from test1

 

 

select * from test2

 

 

select t2.*,t1.market_place_id from test2 t2 
left join test1 t1
on t2.parent_asin=t1.asin


 对照上面例子解释一下这个结果:以左表test2为基准,用右表test1的asin字段和test2的parent_asin字段进行匹配,取出test2的全部数据和test1的market_place_id字段。对于test2中的第一条记录,因为右表中有两条记录符合的asin='parent1',只是market_place_id不同而已(分别为1、2)。于是这两条记录都会作为符合条件的记录加入结果集。这时,虽然是以左表为基准,但是这条记录却在结果集中产生了两条对应的记录。这点要稍加注意:以左表为基准并不意味着结果集的记录数量=左表的记录数量!

 

再回过头来看结果集的5条记录,由ID字段可以很好的区分出每条记录是由左表的哪条记录对应产生的。这里,最后两条记录可以很好的体现出LEFT JOIN的特点。

 

右表有滤条件的LEFT JOIN

这里,我们忽略左表有过滤条件场景的讨论,因为在LEFT JOIN中左表作为基准表,对他的过滤直接反应在SQL的WHERE字句中,效果上也相当于单表SELECT的WHERE字句过滤,缩小左表范围后,再和右表做JOIN,没什么悬念。

但是对于右表的过滤,通常有两种主要的方式:在ON字句中加入过滤条件或者在LEFT JOIN之后的WHERE字句中加入过滤条件。对于这两种方式的对比,下面主要针对逻辑语义和实现性能上加以对比。

 

  • 过滤条件在ON字句中

 

select t2.*,t1.market_place_id from test2 t2 
left join test1 t1
on t2.parent_asin=t1.asin and t1.market_place_id='2'
 上面这条SQL加上了对右表test1中market_place_id的过滤条件:只关心market_place_id为‘2’的右表记录。查询结果如下。 

 



 

逻辑语义上,这个结果相当于右表test1首先进行了条件过滤,只剩下两条记录[(2,'parent1','2'),(3,'parent2','2')],然后左表test2和这个过滤之后的结果集进行无过滤条件的LEFT JOIN,于是得到了上图的结果。

性能上,来看一下这条语句的执行计划截图

 可以看出,T1确实先以2为标准对market_place_id做了一次过滤,然后,在外层,再做原来的LEFT JOIN。由此可以证实上面逻辑语义结果的展示,同时也可以发现,就本例而言,如果能够在market_place_id上建立index,可以直接避免内层过滤对右表进行的全表扫描,从而提高整个SQL的执行效率。下图为在market_place_id上建立index之后,同样SQL语句的执行计划:

 这里可以看出,原来的TABLE FULL SCAN 已经被换成了INDEX的RANGE SCAN,从而也直接导致了Oracle的优化器在最外层的Hash Join替换为了Nested Loops。(当然这个join的方式并不能说明什么问题,因为毕竟测试用的数据集太小,完全有可能在大数据集的真实情况下,优化器根据统计信息还是最终使用Hash Join算法)

  • 过滤条件在WHERE子句中
select t2.*,t1.market_place_id from test2 t2 
left join test1 t1
on t2.parent_asin=t1.asin 
where t1.market_place_id='2'
 上面语句的执行结果如下:

 
逻辑语义上,所有的market_place_id1!='2'的记录(包括NULL)全部被过滤掉了。
性能上,再来看一下这条语句的执行计划:

 
由上面的执行计划可以看出,Oracle也是首先对右表test1进行了market_place_id的过滤,但是过滤之后JOIN操作已经不是LEFT JOIN了,而是变成了普通的INNER JOIN。这就解释了为什么最后的结果集只有两条记录。
同样思路,就本例而言,在右表test的market_place_id字段上建立INDEX,同样可以达到优化SQL的目的,以下是建立INDEX之后的SQL执行计划:
结论:
在使用LEFT JOIN时,右表的限制条件,在ON和WHERE字句中出现,逻辑上的语义完全不同
过滤条件在ON子句中出现时,不会改变原来LEFT JOIN的执行语义:以左表为基表。
过滤条件在WHERE字句中出现时,已经改变了原来LEFT JOIN的语义,相当于在最后LEFT JOIN的结果集里面再做了一次WHERE条件的过滤,所以已经丧失的LEFT JOIN的原始语义。
性能上,其实两者并没有本质的区别,扫描路径完全一致,只是对于后者,Oracle的内部实现,巧妙的将上面描述的语义转换为了通过INNER JOIN实现。这样就保证了在真正执行时还是首先进行内层过滤,缩小右表的数据集,然后进行外层INNER JOIN。
所以使用LEFT JOIN是,有需求对右表进行过滤时,要格外小心了。
备注:
以上测试使用Oracle 11g,更老版本的优化器的执行计划可能会不同。但最终语义上不会有差别。
  • 大小: 2.6 KB
  • 大小: 2.8 KB
  • 大小: 3.7 KB
  • 大小: 3.5 KB
  • 大小: 2.8 KB
  • 大小: 11.8 KB
  • 大小: 12.1 KB
  • 大小: 14.7 KB
  • 大小: 15.6 KB
分享到:
评论
5 楼 ichenwenjin 2016-02-28  
解决了我的问题, 3q
4 楼 yu_duo 2015-03-28  
好仔细的文,正愁这问题。很好的解释。
3 楼 fantasy0407 2015-01-21  
你好,能问下你的查看SQL执行顺序的工具是什么工具。
2 楼 hittyt 2013-08-17  
hitdujuan 写道
昨天在mysql上也偶然遇到这个问题 ,不过跟你场景不一样。
我是要多个条件left join,以前不知道 on后可以加个条件,后来尝试性在on后加了Join的两个条件。
我理解的left join 应该是先确定范围;
where是在这个范围内过滤。(纯属YY)

SQL本身的语法结构上应该都是允许在on后面加上条件的,不过Mysql的具体实现是否也跟Oracle的一致就没研究过了。呵呵。
1 楼 hitdujuan 2013-08-02  
昨天在mysql上也偶然遇到这个问题 ,不过跟你场景不一样。
我是要多个条件left join,以前不知道 on后可以加个条件,后来尝试性在on后加了Join的两个条件。
我理解的left join 应该是先确定范围;
where是在这个范围内过滤。(纯属YY)

相关推荐

    深入Oracle的left join中on和where的区别详解

    在LEFT JOIN中,ON条件用于确定哪些左表的记录与右表的记录相关联。例如,假设我们有两张表,表1(tab1)和表2(tab2),我们可以执行以下查询: 1. `SELECT * FROM tab1 LEFT JOIN tab2 ON (tab1.size = tab2.size...

    oracle中left join和right join的区别浅谈

    在Oracle数据库中,JOIN操作是用于合并两个或更多表中的数据行,以便基于它们之间的相关列获取结果集。LEFT JOIN(也称为LEFT OUTER JOIN)和RIGHT JOIN(也称为RIGHT OUTER JOIN)是两种特殊的JOIN类型,它们在处理...

    left join 过滤条件写在on后面和写在where 后面的区别

    - **LEFT JOIN (或 LEFT OUTER JOIN)**:除了返回两个表中匹配的行外,还会返回左表中不匹配的所有行,并将右表的列设置为NULL。 - **RIGHT JOIN (或 RIGHT OUTER JOIN)**:除了返回两个表中匹配的行外,还会返回右...

    oracle性能优化技巧

    - 将最限制性的条件放在WHERE子句的前面,可以更早地过滤掉不匹配的记录。 2. **避免使用SELECT ***: - 使用具体的列名代替`SELECT *`可以减少数据传输量,从而提高查询速度。 3. **使用解码函数**: - 使用`...

    oracle-join用法

    在Oracle SQL中,Join主要分为几个类型,包括INNER JOIN、LEFT JOIN (LEFT OUTER JOIN)、RIGHT JOIN (RIGHT OUTER JOIN)以及FULL JOIN (FULL OUTER JOIN)。这些类型在Java编程中同样可以使用,通过SQL查询语句来实现...

    left-right-join.zip_join_oracle

    本主题主要聚焦于Oracle中的两种特殊连接方式:左外连接(LEFT JOIN)和右外连接(RIGHT JOIN)。通过深入理解这两种连接类型,我们可以更好地进行数据融合,获取更全面的信息。 首先,让我们从基本概念出发。在SQL...

    oracle 左连接 右连接学习

    Oracle 中的连接(Join)是一种强大的工具,用于将两个或多个表合并成一个结果集。在学习 Oracle 连接时,可能会感到困惑,因为有多种类型的连接,每种类型都有其特点和应用场景。下面我们将详细介绍 Oracle 中的左...

    oracle 资料

    - RIGHT JOIN (或 RIGHT OUTER JOIN):与LEFT JOIN相反,返回右表的所有行,左表中没有匹配的行以NULL填充。 - FULL JOIN (或 FULL OUTER JOIN):返回所有左右表的行,即使在其中一个表中没有匹配的行。没有匹配的行...

    Oracle 左连接、右连接

    本篇文章将深入探讨Oracle中的左连接(LEFT JOIN)和右连接(RIGHT JOIN),这两种连接方式都是为了在不同表之间建立联系,特别是在数据不完全匹配的情况下。 首先,我们来看一下左连接(LEFT JOIN)。左连接返回左...

    oracle数据库表左连接右连接全连接的认识

    Oracle 数据库表连接(Join)是指从两个或多个表中检索数据,并且将它们组合成一个结果集的操作。连接操作可以大大提高数据库查询的效率,特别是在需要访问多个表中的字段时。 连接的基本结构 连接的基本结构是一...

    Oracle左连接返回多条记录中一条记录的查询语句

    Oracle左连接返回多条记录中一条记录的查询语句,更具指定条件分组排序,返回各组中第一条记录

    oracle左右连接方法

    本文将详细介绍Oracle中的左连接(Left Join)、右连接(Right Join)、内连接(Inner Join)、全连接(Full Join)以及交叉连接(Cross Join)的方法,并通过示例来帮助理解每种连接的特点。 ### 一、左连接(Left...

    oracle 连接 join at

    2. **左连接(LEFT JOIN)**:返回左表的所有记录,即使右表中没有匹配的记录。如果右表中没有匹配,则结果为NULL。 3. **右连接(RIGHT JOIN)**:与左连接相反,返回右表的所有记录,即使左表中没有匹配的记录。 ...

    如何解决Oracle分页查询中排序与效率问题

    Oracle 分页查询是指在查询结果中,通过限制行数来实现分页的功能。常用的分页查询方法有两种:ROWNUM 和 OFFSET Fetch。ROWNUM 是 Oracle 专门用于分页查询的伪列,OFFSET Fetch 是 Oracle 12c 及更高版本中引入的...

    oracle 多表查询

    在 Oracle 中,多表查询可以使用不同的连接类型,包括内连接(inner join)、左连接(left join)、右连接(right join)、全连接(full join)和交叉连接(cross join)。 1. 内连接(inner join) 内连接是指...

    Oracle左连接,右连接.doc

    - **左连接(LEFT OUTER JOIN)**:返回左表所有记录以及右表中匹配的记录,右表中没有匹配的记录则填充NULL。 - **右连接(RIGHT OUTER JOIN)**:返回右表所有记录以及左表中匹配的记录,左表中没有匹配的记录则...

    ORACLE表连接方式

    本文将对Oracle中的几种主要表连接方式进行详细介绍:内连接(Inner Join)、自然连接(Natural Join)、左外连接(Left Outer Join)、右外连接(Right Outer Join)、笛卡尔连接(Cartesian Join)、索引连接...

Global site tag (gtag.js) - Google Analytics