1. 程式人生 > >SQL語句:left join後面加上where條件

SQL語句:left join後面加上where條件

select a.*,b.*
from table1 a
left join table2 b on b.X=a.X
where XXX

如上SQL:一旦使用了left join,沒有where條件時,左表table1會顯示全部內容

    而使用了where,只有滿足where條件的記錄才會顯示(左表顯示部分或者全部不顯示)

 

一旦加上where條件,則顯示的結果等於inner join ?

原因分析:

資料庫在通過連線兩張或多張表來返回記錄時,都會生成一張中間的臨時表,然後再將這張臨時表返回給使用者;

where條件是在臨時表生成好後,再對臨時表進行過濾的條件;

因此:where 條件加上,已經沒有left join的含義(必須返回左邊表的記錄)了,條件不為真的就全部過濾掉。 

解決方案:

1、where過濾結果作為子查詢,和主表left,如下:

select a.*,tmp.*
from table1 a
left join(
    select a.*,b.*
    from table1 a
    left join table2 b on b.X=a.X
    where XXX
)tmp

很明顯,子查詢語句無論 left join、inner join都沒啥區別了

2、查詢條件放在on後面

select a.*,b.*
from table1 a
left join table2 b on b.X=a.X and XXX

注意:where XXX去掉,改為連結條件on後面的 and XXX

分析:

 on條件是在生成臨時表時使用的條件,它不管on中的條件是否為真,都會返回左邊表中的記錄。

結論:

過濾條件放在 where後面:是先連線然生成臨時查詢結果,然後再篩選

過濾條件放在 on後面:先根據條件過濾篩選,再連,生成臨時查詢結果

 

還是有點不夠清晰吧 ~~!     

下面舉個栗子-----------------------------

  假設有兩張表:

表1 tab1:
  表2 tab2:
id size size  name
1 10 10 AAA
2 20 20 BBB
3 30 20 CCC

兩條SQL:

第一條SQL的過程:

select * formtab1 left join tab2 on (tab1.size = tab2.size) where tab2.name=’AAA’

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 BBB
3 30 null null

2、再對中間表過濾

where 條件:tab2.name=’AAA’

tab1.id tab1.size  tab2.size tab2.name
1 10 10 AAA

 

第二條SQL的過程:

select * formtab1 left join tab2 on (tab1.size = tab2.size and tab2.name=’AAA’)

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中,返回的結果集是相同的。