原博文地址:http://blog.csdn.net/xingzhemoluo/article/details/39677891

原先一直对SQL左右连接中的on and和on where的区别不是太了解,直到在网上看到了下面这段话才豁然开朗。

在使用left join时,on and和on where条件的区别如下:  
1、on条件是在生成临时表时使用的条件,它不管on中的条件是否为真,都会返回左边表中的记录。  
2、where条件是在临时表生成好后,再对临时表进行过滤的条件。这时已经没有left join的含义(必须返回左边表的记录)了,条件不为真的就全部过滤掉,on后的条件用来生成左右表关联的临时表,where后的条件对临时表中的记录进行过滤。

实践是检验真理的唯一标准,接下来用实验来证明一下:

先上表结构以及数据:

  1. SQL> create table A (id int, type int);
  2. SQL> select * from A;
  3. ID       TYPE
  4. ---------- ----------
  5. 1          1
  6. 2          1
  7. 3          2
  8. SQL> create table B(id int ,class int);
  9. SQL> select * from B;
  10. ID      CLASS
  11. ---------- ----------
  12. 1          1
  13. 2          2

  1. SQL> select * from A left join B on A.id = B.id where A.type = 1;
  2. ID       TYPE         ID      CLASS
  3. ---------- ---------- ---------- ----------
  4. 1          1          1          1
  5. 2          1          2          2

根据上面那段话的解释,where字句是在生成临时表以后再进行过滤的,也就是可以理解为就是一个左连接:select * from A left join B on A.id = B.id;

其运行结果如下:

  1. SQL> select * from A left join B on A.id = b.id;
  2. ID       TYPE         ID      CLASS
  3. ---------- ---------- ---------- ----------
  4. 1          1          1          1
  5. 2          1          2          2
  6. 3          2

然后加上where A.type = 1对临时表进行过滤,除掉A.type不为1的,显然结果正确。

  1. SQL> select * from A left join B on A.id = B.id and A.type = 1;
  2. ID       TYPE         ID      CLASS
  3. ---------- ---------- ---------- ----------
  4. 1          1          1          1
  5. 2          1          2          2
  6. 3          2

因为左连接不管on and语句是否为真都必须返回左表所有的记录,所以and A.type=1;没有起到任何作用。

  1. SQL> select * from A left join B on A.id = B.id and B.class = 1;
  2. ID       TYPE         ID      CLASS
  3. ---------- ---------- ---------- ----------
  4. 1          1          1          1
  5. 3          2
  6. 2          1

根据上面那段话的解释:on条件是在生成临时表时使用的条件,它不管on中的条件是否为真,都会返回左边表中的记录。显然左连接再加上新的条件:B.class = 1筛选掉第二行记录,结果正确。

  1. SQL> select * from A left join B on A.id = B.id where B.class = 1;
  2. ID       TYPE         ID      CLASS
  3. ---------- ---------- ---------- ----------
  4. 1          1          1          1

原因通①。(①,②)和(③,④)就是关键词on and和on where的区别,但结果却完全不同。综上三个例子,where是生成临时表以后再进行过滤,对左右表都进行筛选。而and后面的语句如果是对left join中的左表进行过滤将不起任何作用,对右表进行过滤的话,那么左表还是返回所有行,只是右表会被过滤掉一部分行。

再来看看内连接inner join  on and和 on where的区别:

由于刚开始表的数据不是太适合,所以先稍微更新一下,这样更好观察inner join和left join在and和where的不同之处。

  1. SQL> update A set type = 2 where id = 1;
  2. 已更新 1 行。
  3. SQL> select * from A;
  4. ID       TYPE
  5. ---------- ----------
  6. 1          2
  7. 2          1
  8. 3          2
  9. SQL> select * from B;
  10. ID      CLASS
  11. ---------- ----------
  12. 1          1
  13. 2          2

先看看and的:

  1. SQL> select * from A inner join B on A.id = B.id and A.type = 1;
  2. ID       TYPE         ID      CLASS
  3. ---------- ---------- ---------- ----------
  4. 2          1          2          2

显然输出结果与左连接的不一样,先与没有and的内连接比较一下:

  1. SQL> select * from A inner join B on A.id = B.id;
  2. ID       TYPE         ID      CLASS
  3. ---------- ---------- ---------- ----------
  4. 1          2          1          1
  5. 2          1          2          2

显然如果按左连接的逻辑,这个结果就是错误的。但这是Oracle输出的,而不是我瞎打的,显然在内连接时与左连接不同了。这里on and条件和on where条件一样对生成以后的临时表同样会被过滤。显然A表id为1的type不为1,所以它被过滤了。

再上几组来验证一下上面这个猜想是否是正确的:

再来看看内连接的where:

  1. SQL> select * from A inner join B on A.id = B.id where A.type = 1;
  2. ID       TYPE         ID      CLASS
  3. ---------- ---------- ---------- ----------
  4. 2          1          2          2

这个也是和左连接一样生成临时表然后进行过滤,不作解释。

  1. SQL> select * from A inner join B on A.id = B.id and B.class = 1;
  2. ID       TYPE         ID      CLASS
  3. ---------- ---------- ---------- ----------
  4. 1          2          1          1

  1. SQL> select * from A inner join B on A.id = B.id where B.class = 1;
  2. ID       TYPE         ID      CLASS
  3. ---------- ---------- ---------- ----------
  4. 1          2          1          1

对比发现:(⑤,⑥)和(⑦,⑧)结果都一样,也就是说内连接inner join on and 或者on where不管是对左表还是右表进行过滤,实际都是在生成临时表以后再进行过滤的,而且对左表和右表都起作用,这与左连接left join有本质的区别!!!

最后上俩没有使用连接语句的例子:

  1. </pre><p></p><pre name="code" class="sql">SQL> select * from A,B where A.id = B.id and A.type = 1;
  2. ID       TYPE         ID      CLASS
  3. ---------- ---------- ---------- ----------
  4. 1          1          1          1
  5. 2          1          2          2
  6. SQL> select * from A,B where A.type = 1 and A.id = B.id;
  7. ID       TYPE         ID      CLASS
  8. ---------- ---------- ---------- ----------
  9. 1          1          1          1
  10. 2          1          2          2

比较简单,不作解释。

总结一下:

        在使用left join时,on和where条件的区别如下:  
1、on条件是在生成临时表时使用的条件,它不管on中的条件是否为真,都会返回左边表中的记录。(实际上左连接中如果and语句是对左表进行过滤的,那么不管真假都不起任何作用。如果是对右表过滤的,那么左表所有记录都返回,右表筛选以后再与左表连接返回)  
2、where条件是在临时表生成好后,再对临时表进行过滤的条件。这时已经没有left join的含义(必须返回左边表的记录)了,条件不为真的就全部过滤掉,on后的条件用来生成左右表关联的临时表,where后的条件对临时表中的记录进行过滤。

        在使用inner join时,不管是对左表还是右表进行筛选,on and和on where都会对生成的临时表进行过滤。

转!!SQL左右连接中的on and和on where的区别的更多相关文章

  1. SQL左右连接中的on and和on where的区别

    SQL左右连接中的on and和on where的区别 左联时,ON后面的对左边表的条件对左边表数据无影响(因为左连接符合左边所有条件),但对右边表数据有影响,只有符合左边表条件时,右边表数据才会查出 ...

  2. SQL Server连接查询之Cross Apply和Outer Apply的区别及用法(转载)

    先简单了解下cross apply的语法以及会产生什么样的结果集吧!示例表: SELECT * FROM tableA CROSS APPLY tableB 两张表直接连接,不需要任何的关联条件,产生 ...

  3. sql server存储过程中SELECT 与 SET 对变量赋值的区别

    SQL Server 中对已经定义的变量赋值的方式用两种,分别是 SET 和 SELECT. 对于这两种方式的区别,SQL Server 联机丛书中已经有详细的说明,但很多时候我们 并没有注意,其实这 ...

  4. 【转】sql server存储过程中SELECT 与 SET 对变量赋值的区别

    转自:http://www.cnblogs.com/micheng11/archive/2008/07/08/1237905.html SQL Server 中对已经定义的变量赋值的方式用两种,分别是 ...

  5. sql server存储过程中SELECT 与 SET 对变量赋值的区别 转自Theo

    SQL Server 中对已经定义的变量赋值的方式用两种,分别是 SET 和 SELECT. 对于这两种方式的区别,SQL Server 联机丛书中已经有详细的说明,但很多时候我们 并没有注意,其实这 ...

  6. 配置sql server 2000以允许远程访问 及 连接中的四个最常见错误

    地址:http://www.cnblogs.com/JoshuaDreaming/archive/2010/12/01/1893242.html 配置sql server 2000以允许远程访问适合故 ...

  7. 利用Ring Buffer在SQL Server 2008中进行连接故障排除

    原文:利用Ring Buffer在SQL Server 2008中进行连接故障排除 出自:http://blogs.msdn.com/b/apgcdsd/archive/2011/11/21/ring ...

  8. 通过sql server 连接mysql

    图文:通过sql server 连接mysql   1.在SQL SERVER服务器上安装MYSQL ODBC驱动; 驱动下载地址:http://dev.mysql.com/downloads/con ...

  9. SQL Server 连接问题案例解析(1)

    SQL Server 连接问题案例解析(1) 转载自:http://blogs.msdn.com/b/apgcdsd/archive/2015/04/27/sql.aspx?CommentPosted ...

随机推荐

  1. Atitti.数据操作crud js sdk dataServiceV3设计说明

    Atitti.数据操作crud js sdk dataServiceV3设计说明 1. 增加数据1 1.1. 参数哦说明1 2. 查询数据1 2.1. 参数说明2 3. 更新数据2 3.1. 参数说明 ...

  2. 李洪强iOS开发之iOS学习方法收集

    李洪强iOS开发之iOS学习方法收集 在这里收集一些iOS学习方法,会不断更新 项目 简述 日期 一年多iOS开发总结 作者总结了自己一年多的iOS学习经验,对于iOS初学者来说很多地方是可以借鉴的 ...

  3. jQuery 时间戳转化成时间

    //时间转换  function   formatDate(now)   { var   now= new Date(now);      var   year=now.getFullYear();  ...

  4. android camera之nv21旋转

    这周做的一个android的camera开发,需要获取到视频帧数据,并且需要是nv21格式的byte数组,并且视频帧的图像需要是正方向的.和android相机打过交道的都清楚,android的came ...

  5. 如何执行静默(无人参与)Java 安装?

    http://www.java.com/zh_CN/download/help/silent_install.xml

  6. 01 SQL核心语句

    在 sqlplus 中可以使用 ; 或 / 来执行一条 sql 语句, 但是 / 必须要换行, 个人感觉只有那些脚本啊什么的最后一般使用 / , 其他的都使用; 核心语句 SELECT oracle  ...

  7. 第二百四十一节,Bootstrap进度条媒体对象和 Well 组件

    第二百四十一节,Bootstrap进度条媒体对象和 Well 组件 学习要点: 1.Well 组件 2.进度条组件 3.媒体对象组件 本节课我们主要学习一下 Bootstrap 的三个组件功能:Wel ...

  8. python 数据提取之JSON与JsonPATH

    JSON(JavaScript Object Notation) 是一种轻量级的数据交换格式,它使得人们很容易的进行阅读和编写.同时也方便了机器进行解析和生成.适用于进行数据交互的场景,比如网站前台与 ...

  9. yalmip + lpsolve + matlab 求解混合整数线性规划问题(MIP/MILP)

    最近建立了一个网络流模型,是一个混合整数线性规划问题(模型中既有连续变量,又有整型变量).当要求解此模型的时候,发现matlab优化工具箱竟没有自带的可以求解这类问题的算法(只有bintprog求解器 ...

  10. scrollHeight、clientHeight、offsetHeight、scrollTop等的定义以及图解

    开发中经常遇到使用scrollHeight.scrollTop.clientHeight.offsetHeight等的情况,网上有众多关于这些属性的解释,但是并不全面和直观,现在将这些属性结合图例整理 ...