IN&EXISTS與NOT IN&NOT EXISTS 的優(yōu)化原則小結(jié)
更新時(shí)間:2010年06月21日 11:57:50 作者:
下面只是從理論上提出了一些建議,最好的原則是大家在上面的基礎(chǔ)上,能夠使用執(zhí)行計(jì)劃來(lái)分析,得出最佳的語(yǔ)句的寫法。
1. EXISTS的執(zhí)行流程
select * from t1 where exists ( select null from t2 where y = x )
可以理解為:
for x in ( select * from t1 )
loop
if ( exists ( select null from t2 where y = x.x )
then
OUTPUT THE RECORD
end if
end loop
對(duì)于in 和 exists的性能區(qū)別:
如果子查詢得出的結(jié)果集記錄較少,主查詢中的表較大且又有索引時(shí)應(yīng)該用in,反之如果外層的主查詢記錄較少,子查詢中的表大,又有索引時(shí)使用exists。
其實(shí)我們區(qū)分in和exists主要是造成了驅(qū)動(dòng)順序的改變(這是性能變化的關(guān)鍵),如果是exists,那么以外層表為驅(qū)動(dòng)表,先被訪問,如果是IN,那么先執(zhí)行子查詢,所以我們會(huì)以驅(qū)動(dòng)表的快速返回為目標(biāo),那么就會(huì)考慮到索引及結(jié)果集的關(guān)系了
另外IN時(shí)不對(duì)NULL進(jìn)行處理,如:
select 1 from dual where null in (0,1,2,null)
結(jié)果為空。
2. NOT IN 與NOT EXISTS:
NOT EXISTS的執(zhí)行流程
select .....
from rollup R
where not exists ( select 'Found' from title T
where R.source_id = T.Title_ID);
可以理解為:
for x in ( select * from rollup )
loop
if ( not exists ( that query ) ) then
OUTPUT
end if;
end;
注意:NOT EXISTS 與 NOT IN 不能完全互相替換,看具體的需求。如果選擇的列可以為空,則不能被替換。
例如下面語(yǔ)句,看他們的區(qū)別:
select x,y from t;
x y
------ ------
1 3
3 1
1 2
1 1
3 1
5
select * from t where x not in (select y from t t2 )
no rows
select * from t where not exists (select null from t t2
where t2.y=t.x )
x y
------ ------
5 NULL
所以要具體需求來(lái)決定
對(duì)于not in 和 not exists的性能區(qū)別:
not in 只有當(dāng)子查詢中,select 關(guān)鍵字后的字段有not null約束或者有這種暗示時(shí)用not in,另外如果主查詢中表大,子查詢中的表小但是記錄多,則應(yīng)當(dāng)使用not in,并使用anti hash join.
如果主查詢表中記錄少,子查詢表中記錄多,并有索引,可以使用not exists,另外not in最好也可以用/*+ HASH_AJ */或者外連接+is null
NOT IN 在基于成本的應(yīng)用中較好
比如:
select .....
from rollup R
where not exists ( select 'Found' from title T
where R.source_id = T.Title_ID);
改成(佳)
select ......
from title T, rollup R
where R.source_id = T.Title_id(+)
and T.Title_id is null;
或者(佳)
sql> select /*+ HASH_AJ */ ...
from rollup R
where ource_id NOT IN ( select ource_id
from title T
where ource_id IS NOT NULL )
注意:上面只是從理論上提出了一些建議,最好的原則是大家在上面的基礎(chǔ)上,能夠使用執(zhí)行計(jì)劃來(lái)分析,得出最佳的語(yǔ)句的寫法。
'//=============================
exists,not exists總結(jié)
1 exists
SELECT * FROM anken_m WHERE EXISTS(
SELECT my_list_temp_m.sales_code
FROM my_list_temp_m
WHERE my_list_temp_m.sales_code=anken_m.sales_code)
說明:
1) 查詢?cè)赼nken_m表和my_list_temp_m表中都存在的sales_code。
2) sales_code是anken_m的主鍵,my_list_temp_m的外鍵。
注意:
1) 外層查詢表anken_m是查詢的對(duì)象。
2) 內(nèi)層查詢表my_list_temp_m是條件對(duì)象。
3) 內(nèi)外層的查詢表不能相同。
4) 作為關(guān)聯(lián)條件的anken_m表不需要在內(nèi)層查詢FROM后添加。
5) my_list_temp_m.sales_code=anken_m.sales_code條件的左右順序不影響查詢結(jié)果。
2 not exists
SELECT * FROM anken_m WHERE NOT EXISTS(
SELECT my_list_temp_m.sales_code
FROM my_list_temp_m
WHERE my_list_temp_m.sales_code=anken_m.sales_code)
說明:
1) 查詢?cè)赼nken_m表中存在,但是在my_list_temp_m表中不存在的sales_code。
2) sales_code是anken_m的主鍵,my_list_temp_m的外鍵。
注意:
1) 外層查詢表anken_m是查詢的對(duì)象。
2) 內(nèi)層查詢表my_list_temp_m是條件對(duì)象。
3) 內(nèi)外層的查詢表不能相同。
4) 作為關(guān)聯(lián)條件的anken_m表不需要在內(nèi)層查詢FROM后添加。
5) my_list_temp_m.sales_code=anken_m.sales_code條件的左右順序不影響查詢結(jié)果。
3 綜合運(yùn)用
UPDATE anken_m
SET(plan_type_code, branch_name, business_type_code)
=(SELECT anken.plan_type_code,anken.branch_name,anken.business_type_code
FROM anken
WHERE anken.sales_code=anken_m.sales_code)
WHERE EXISTS (
SELECT anken.sales_code
FROM anken,my_list_temp_m
WHERE my_list_temp_m.sales_code=anken.sales_code
AND anken.sales_code=anken_m.sales_code
)
說明:
1) 用一個(gè)表的記錄數(shù)據(jù)更新另一個(gè)表的記錄數(shù)據(jù)。
2) 用一個(gè)SQL語(yǔ)句進(jìn)行批量更新。
2) sales_code是anken,anken_m的主鍵,my_list_temp_m的外鍵。
注意:
1) set 語(yǔ)句中的要被更新字段必須跟數(shù)據(jù)源字段一一對(duì)應(yīng),另外數(shù)據(jù)源查詢中的條件必須限定一條記錄。也就是根據(jù)sales_code可以唯一確定anken的一條記錄,和anken_m的一條記錄,這樣才能保證要被更新的記錄和數(shù)據(jù)源記錄的主鍵是相同的。
2) 根據(jù)WHERE EXISTS語(yǔ)句可以確定數(shù)據(jù)源記錄的范圍,也就是可以用anken表中哪些記錄更新anken_m表。所以anken_m不需要在WHERE EXISTS語(yǔ)句中的FROM后添加。
select * from t1 where exists ( select null from t2 where y = x )
可以理解為:
復(fù)制代碼 代碼如下:
for x in ( select * from t1 )
loop
if ( exists ( select null from t2 where y = x.x )
then
OUTPUT THE RECORD
end if
end loop
對(duì)于in 和 exists的性能區(qū)別:
如果子查詢得出的結(jié)果集記錄較少,主查詢中的表較大且又有索引時(shí)應(yīng)該用in,反之如果外層的主查詢記錄較少,子查詢中的表大,又有索引時(shí)使用exists。
其實(shí)我們區(qū)分in和exists主要是造成了驅(qū)動(dòng)順序的改變(這是性能變化的關(guān)鍵),如果是exists,那么以外層表為驅(qū)動(dòng)表,先被訪問,如果是IN,那么先執(zhí)行子查詢,所以我們會(huì)以驅(qū)動(dòng)表的快速返回為目標(biāo),那么就會(huì)考慮到索引及結(jié)果集的關(guān)系了
另外IN時(shí)不對(duì)NULL進(jìn)行處理,如:
select 1 from dual where null in (0,1,2,null)
結(jié)果為空。
2. NOT IN 與NOT EXISTS:
NOT EXISTS的執(zhí)行流程
復(fù)制代碼 代碼如下:
select .....
from rollup R
where not exists ( select 'Found' from title T
where R.source_id = T.Title_ID);
可以理解為:
復(fù)制代碼 代碼如下:
for x in ( select * from rollup )
loop
if ( not exists ( that query ) ) then
OUTPUT
end if;
end;
注意:NOT EXISTS 與 NOT IN 不能完全互相替換,看具體的需求。如果選擇的列可以為空,則不能被替換。
例如下面語(yǔ)句,看他們的區(qū)別:
select x,y from t;
x y
------ ------
1 3
3 1
1 2
1 1
3 1
5
select * from t where x not in (select y from t t2 )
no rows
select * from t where not exists (select null from t t2
where t2.y=t.x )
x y
------ ------
5 NULL
所以要具體需求來(lái)決定
對(duì)于not in 和 not exists的性能區(qū)別:
not in 只有當(dāng)子查詢中,select 關(guān)鍵字后的字段有not null約束或者有這種暗示時(shí)用not in,另外如果主查詢中表大,子查詢中的表小但是記錄多,則應(yīng)當(dāng)使用not in,并使用anti hash join.
如果主查詢表中記錄少,子查詢表中記錄多,并有索引,可以使用not exists,另外not in最好也可以用/*+ HASH_AJ */或者外連接+is null
NOT IN 在基于成本的應(yīng)用中較好
比如:
復(fù)制代碼 代碼如下:
select .....
from rollup R
where not exists ( select 'Found' from title T
where R.source_id = T.Title_ID);
改成(佳)
select ......
from title T, rollup R
where R.source_id = T.Title_id(+)
and T.Title_id is null;
或者(佳)
sql> select /*+ HASH_AJ */ ...
from rollup R
where ource_id NOT IN ( select ource_id
from title T
where ource_id IS NOT NULL )
注意:上面只是從理論上提出了一些建議,最好的原則是大家在上面的基礎(chǔ)上,能夠使用執(zhí)行計(jì)劃來(lái)分析,得出最佳的語(yǔ)句的寫法。
'//=============================
exists,not exists總結(jié)
1 exists
SELECT * FROM anken_m WHERE EXISTS(
SELECT my_list_temp_m.sales_code
FROM my_list_temp_m
WHERE my_list_temp_m.sales_code=anken_m.sales_code)
說明:
1) 查詢?cè)赼nken_m表和my_list_temp_m表中都存在的sales_code。
2) sales_code是anken_m的主鍵,my_list_temp_m的外鍵。
注意:
1) 外層查詢表anken_m是查詢的對(duì)象。
2) 內(nèi)層查詢表my_list_temp_m是條件對(duì)象。
3) 內(nèi)外層的查詢表不能相同。
4) 作為關(guān)聯(lián)條件的anken_m表不需要在內(nèi)層查詢FROM后添加。
5) my_list_temp_m.sales_code=anken_m.sales_code條件的左右順序不影響查詢結(jié)果。
2 not exists
SELECT * FROM anken_m WHERE NOT EXISTS(
SELECT my_list_temp_m.sales_code
FROM my_list_temp_m
WHERE my_list_temp_m.sales_code=anken_m.sales_code)
說明:
1) 查詢?cè)赼nken_m表中存在,但是在my_list_temp_m表中不存在的sales_code。
2) sales_code是anken_m的主鍵,my_list_temp_m的外鍵。
注意:
1) 外層查詢表anken_m是查詢的對(duì)象。
2) 內(nèi)層查詢表my_list_temp_m是條件對(duì)象。
3) 內(nèi)外層的查詢表不能相同。
4) 作為關(guān)聯(lián)條件的anken_m表不需要在內(nèi)層查詢FROM后添加。
5) my_list_temp_m.sales_code=anken_m.sales_code條件的左右順序不影響查詢結(jié)果。
3 綜合運(yùn)用
UPDATE anken_m
SET(plan_type_code, branch_name, business_type_code)
=(SELECT anken.plan_type_code,anken.branch_name,anken.business_type_code
FROM anken
WHERE anken.sales_code=anken_m.sales_code)
WHERE EXISTS (
SELECT anken.sales_code
FROM anken,my_list_temp_m
WHERE my_list_temp_m.sales_code=anken.sales_code
AND anken.sales_code=anken_m.sales_code
)
說明:
1) 用一個(gè)表的記錄數(shù)據(jù)更新另一個(gè)表的記錄數(shù)據(jù)。
2) 用一個(gè)SQL語(yǔ)句進(jìn)行批量更新。
2) sales_code是anken,anken_m的主鍵,my_list_temp_m的外鍵。
注意:
1) set 語(yǔ)句中的要被更新字段必須跟數(shù)據(jù)源字段一一對(duì)應(yīng),另外數(shù)據(jù)源查詢中的條件必須限定一條記錄。也就是根據(jù)sales_code可以唯一確定anken的一條記錄,和anken_m的一條記錄,這樣才能保證要被更新的記錄和數(shù)據(jù)源記錄的主鍵是相同的。
2) 根據(jù)WHERE EXISTS語(yǔ)句可以確定數(shù)據(jù)源記錄的范圍,也就是可以用anken表中哪些記錄更新anken_m表。所以anken_m不需要在WHERE EXISTS語(yǔ)句中的FROM后添加。
相關(guān)文章
SQLServer數(shù)據(jù)庫(kù)游標(biāo)的具體使用
本文主要介紹了SQLServer數(shù)據(jù)庫(kù)游標(biāo)的具體使用,游標(biāo)通常用于需要逐行處理數(shù)據(jù)的情況,例如對(duì)每一行數(shù)據(jù)進(jìn)行特定的計(jì)算或更新操作,下面就一起來(lái)了解一下2024-08-08
揭秘SQL Server 2014有哪些新特性(2)-固態(tài)硬盤 Buffer Pool(緩沖池) 擴(kuò)展
SQL Server 2014 中引入的緩沖池?cái)U(kuò)展提供數(shù)據(jù)庫(kù)引擎緩沖池的非易失性隨機(jī)存取內(nèi)存(即固態(tài)硬盤)擴(kuò)展的無(wú)縫集成,從而顯著提高 I/O 吞吐量。 那么今天我們來(lái)探究下,這個(gè)功能到底如何強(qiáng)悍吧2014-08-08
sqlserver 統(tǒng)計(jì)sql語(yǔ)句大全收藏
SQL統(tǒng)計(jì)大全收藏,主要是一些實(shí)現(xiàn)統(tǒng)計(jì)功能常用的代碼,希望對(duì)需要的朋友有所幫助.2010-05-05
SQL Server 2016 CTP2.2安裝配置方法圖文教程
這篇文章主要為大家詳細(xì)介紹了SQL Server 2016 CTP2.2安裝配置方法圖文教程 ,感興趣的小伙伴們可以參考一下2016-07-07
重裝MS SQL Server 2000前必須徹底刪除原安裝文件的方法
重裝MS SQL Server 2000前必須徹底刪除原安裝文件的方法...2007-11-11
詳解SqlServer數(shù)據(jù)庫(kù)中Substring函數(shù)的用法
substring操作的字符串,開始截取的位置,返回的字符個(gè)數(shù),本文通過簡(jiǎn)單實(shí)例給大家介紹了SqlServer數(shù)據(jù)庫(kù)中Substring函數(shù)的用法,感興趣的朋友一起看看吧2018-04-04

