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