用where 是先连接然后再筛选   
用and 是先筛选再连接 


数据库在通过连接两张或多张表来返回记录时,都会生成一张中间的临时表,然后再将这张临时表返回给用户。 

      在使用left jion时,on和where条件的区别如下: 

1、 on条件是在生成临时表时使用的条件,它不管on中的条件是否为真,都会返回左边表中的记录。 

2、where条件是在临时表生成好后,再对临时表进行过滤的条件。这时已经没有left join的含义(必须返回左边表的记录)了,条件不为真的就全部过滤掉。 

       假设有两张表: 

表1 tab1: 

id size 

1 10 

2 20 

3 30 

表2 tab2: 

size name 

10 AAA 

20 BBB 

20 CCC 


两条SQL: 
1、select * form tab1 left join tab2 on (tab1.size = tab2.size) where tab2.name=’AAA’ 
2、select * form tab1 left join tab2 on (tab1.size = tab2.size and tab2.name=’AAA’) 

第一条SQL的过程: 

1、中间表 
on条件: 
tab1.size = tab2.size 

tab1.id    tab1.size    tab2.size     tab2.name 

1               10                   10               AAA 

2              20                     20             BBB 

2             20                      20               CCC 

3             30                    (null)              (null) 

2、再对中间表过滤 
where 条件: 
tab2.name=’AAA’ 

tab1.id       tab1.size        tab2.size     tab2.name 

1                  10                  10              AAA 


第二条SQL的过程: 

1、中间表 
on条件: 
tab1.size = tab2.size and tab2.name=’AAA’ 
(条件不为真也会返回左表中的记录) 

tab1.id      tab1.size         tab2.size       tab2.name 

1               10                     10                   AAA 

2               20                   (null)               (null) 

3               30                    (null)                 (null) 

     其实以上结果的关键原因就是left join,right join,full join的特殊性,不管on上的条件是否为真都会返回left或right表中的记录,full则具有left和right的特性的并集。 而inner jion没这个特殊性,则条件放在on中和where中,返回的结果集是相同的。 
 

on中的条件限制

  
SQL> select * from tab1;
 
        ID      SIZE1
---------- ----------
         1         10
         2         20
         3         30
 
SQL> select * from  tab2;
 
     SIZE1 NAME
---------- --------------------
        10 AAA
        20 BBB
        20 CCC
 
SQL> select tab1.*,tab2.* from tab1 left join tab2 on (tab1.size1= tab2.size1) ;
 
        ID      SIZE1      SIZE1 NAME
---------- ---------- ---------- --------------------
         1         10         10 AAA
         2         20         20 BBB
         2         20         20 CCC
         3         30
 
SQL> select tab1.*,tab2.* from tab1 left join tab2 on (tab1.size1= tab2.size1 and tab1.id=2);

        ID      SIZE1      SIZE1 NAME
---------- ---------- ---------- --------------------
         1         10
         2         20         20 BBB
         2         20         20 CCC
         3         30
 
SQL> select tab1.*,tab2.* from tab1 left join tab2 on (tab1.size1= tab2.size1 and tab2.name='AAA');
 
        ID      SIZE1      SIZE1 NAME
---------- ---------- ---------- --------------------
         1         10         10 AAA
         3         30
         2         20
 
SQL> select tab1.*,tab2.* from tab1 left join tab2 on (tab1.size1= tab2.size1 and tab2.name='BBB');
 
        ID      SIZE1      SIZE1 NAME
---------- ---------- ---------- --------------------
         2         20         20 BBB
         3         30
         1         10
 
SQL> select tab1.*,tab2.* from tab1 left join tab2 on (tab1.size1= tab2.size1 and tab2.name='CCC');
 
        ID      SIZE1      SIZE1 NAME
---------- ---------- ---------- --------------------
         2         20         20 CCC
         3         30
         1         10
 
SQL> update tab2 set name='DDD' where size1=20;
 
已更新2行。
 
提交完成。
SQL> commit
  2  ;
 
提交完成。
 
SQL> select tab1.*,tab2.* from tab1 left join tab2 on (tab1.size1= tab2.size1 and tab2.name='DDD');
 
        ID      SIZE1      SIZE1 NAME
---------- ---------- ---------- --------------------
         2         20         20 DDD
         2         20         20 DDD
         3         30
         1         10
 
SQL> select tab1.*,tab2.* from tab1 left join tab2 on (tab1.size1= tab2.size1 and tab2.name='xxx');
 
        ID      SIZE1      SIZE1 NAME
---------- ---------- ---------- --------------------
         3         30
         2         20
         1         10

当on中对左表的非连接字段限制时 与 对右表的非连接字段限制时 是两种不同的情况,请注意。


当on中对右表的非连接字段限制时(on (tab1.size1= tab2.size1 and tab2.name='AAA')) 相当于右表根据非连接字段限制获取结果,然后左表再与它关联。
select tab1.*,tab2.* from tab1 left join tab2 on (tab1.size1= tab2.size1 and tab2.name='AAA');
相当于
select tab1.*,t.* from tab1 left join (select * from tab2 where tab2.name='AAA') t on (tab1.size1= t.size1);