IN&EXISTS與NOT IN&NOT EXISTS 的優化准繩小結。本站提示廣大學習愛好者:(IN&EXISTS與NOT IN&NOT EXISTS 的優化准繩小結)文章只能為提供參考,不一定能成為您想要的結果。以下是IN&EXISTS與NOT IN&NOT EXISTS 的優化准繩小結正文
1. EXISTS的履行流程
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
關於in 和 exists的機能差別:
假如子查詢得出的成果集記載較少,主查詢中的表較年夜且又有索引時應當用in,反之假如外層的主查詢記載較少,子查詢中的表年夜,又有索引時應用exists。
其實我們辨別in和exists重要是形成了驅動次序的轉變(這是機能變更的症結),假如是exists,那末之外層表為驅動表,先被拜訪,假如是IN,那末先履行子查詢,所以我們會以驅動表的疾速前往為目的,那末就會斟酌到索引及成果集的關系了
別的IN時纰謬NULL停止處置,如:
select 1 from dual where null in (0,1,2,null)
成果為空。
2. NOT IN 與NOT EXISTS:
NOT EXISTS的履行流程
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 不克不及完整相互調換,看詳細的需求。假如選擇的列可認為空,則不克不及被調換。
例以下面語句,看他們的差別:
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
所以要詳細需求來決議
關於not in 和 not exists的機能差別:
not in 只要當子查詢中,select 症結字後的字段有not null束縛或許有這類暗示時用not in,別的假如主查詢中表年夜,子查詢中的表小然則記載多,則應該應用not in,並應用anti hash join.
假如主查詢表中記載少,子查詢表中記載多,並有索引,可使用not exists,別的not in最好也能夠用/*+ HASH_AJ */或許外銜接+is null
NOT IN 在基於本錢的運用中較好
好比:
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 )
留意:下面只是從實際上提出了一些建議,最好的准繩是年夜家在下面的基本上,可以或許應用履行籌劃來剖析,得出最好的語句的寫法。
'//=============================
exists,not exists總結
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) 查詢在anken_m表和my_list_temp_m表中都存在的sales_code。
2) sales_code是anken_m的主鍵,my_list_temp_m的外鍵。
留意:
1) 外層查詢表anken_m是查詢的對象。
2) 內層查詢表my_list_temp_m是前提對象。
3) 表裡層的查詢表不克不及雷同。
4) 作為聯系關系前提的anken_m表不須要在內層查詢FROM後添加。
5) my_list_temp_m.sales_code=anken_m.sales_code前提的閣下次序不影響查詢成果。
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) 查詢在anken_m表中存在,然則在my_list_temp_m表中不存在的sales_code。
2) sales_code是anken_m的主鍵,my_list_temp_m的外鍵。
留意:
1) 外層查詢表anken_m是查詢的對象。
2) 內層查詢表my_list_temp_m是前提對象。
3) 表裡層的查詢表不克不及雷同。
4) 作為聯系關系前提的anken_m表不須要在內層查詢FROM後添加。
5) my_list_temp_m.sales_code=anken_m.sales_code前提的閣下次序不影響查詢成果。
3 綜合應用
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) 用一個表的記載數據更新另外一個表的記載數據。
2) 用一個SQL語句停止批量更新。
2) sales_code是anken,anken_m的主鍵,my_list_temp_m的外鍵。
留意:
1) set 語句中的要被更新字段必需跟數據源字段逐個對應,別的數據源查詢中的前提必需限制一筆記錄。也就是依據sales_code可以獨一肯定anken的一筆記錄,和anken_m的一筆記錄,如許能力包管要被更新的記載和數據源記載的主鍵是雷同的。
2) 依據WHERE EXISTS語句可以肯定數據源記載的規模,也就是可以用anken表中哪些記載更新anken_m表。所以anken_m不須要在WHERE EXISTS語句中的FROM後添加。