(轉)ORACLE 中IN和EXISTS比較
原文:http://www.cnblogs.com/yf520gn/archive/2009/01/12/1374359.html
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,那麼先執行子查詢,所以我們會以驅動表的快速返回為目標,那麼就會考慮到索引及結果集的關係了
如:
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 )
注意:上面只是從理論上提出了一些建議,最好的原則是大家在上面的基礎上,能夠使用執行計劃來分析,得出最佳的語句的寫法
希望大家提出異議
來自 “ ITPUB部落格 ” ,連結:http://blog.itpub.net/124805/viewspace-1028694/,如需轉載,請註明出處,否則將追究法律責任。
相關文章
- ORACLE 中IN和EXISTS比較Oracle
- [Oracle] minus 和 not exists比較Oracle
- oracle中關於in和exists,not in 和 not existsOracle
- (轉)ORACLE 中IN和EXISTS的區別Oracle
- oracle中的exists和not exists和in用法詳解Oracle
- oracle中的exists 和not exists 用法詳解Oracle
- [Oracle] exists 和 not existsOracle
- Oracle date 型別比較和String比較Oracle型別
- oracle中in和exists的區別Oracle
- 查詢oracle比較慢的session和SQL[轉]OracleSessionSQL
- Oracle 中 replace函式和translate函式比較Oracle函式
- Oracle中exists和in的效能差異Oracle
- ORACLE和MSSQL中行鎖比較OracleSQL
- 透過sql trace比較常規 not in 、minus、not exists效率SQL
- 通過sql trace比較常規 not in 、minus、not exists效率SQL
- [轉]Oracle分割槽索引--本地索引和全域性索引比較Oracle索引
- Oracle分割槽索引--本地索引和全域性索引比較(轉)Oracle索引
- 用PHP連mysql和oracle資料庫效能比較(轉)PHPMySqlOracle資料庫
- sqlldr和oracle_datapump效能比較SQLOracle
- 通過shell來比較oracle和java中的字串使用OracleJava字串
- 透過shell來比較oracle和java中的字串使用OracleJava字串
- oracle中的exists理解Oracle
- Oracle與SQL Server在企業應用中的比較(轉)OracleSQLServer
- js 深比較和淺比較JS
- ORACLE DATE和TIMESTAMP資料型別的比較(一) (轉)Oracle資料型別
- ORACLE DATE和TIMESTAMP資料型別的比較(二) (轉)Oracle資料型別
- MySQL中的NULL和空串比較MySqlNull
- SQL中IN,NOT IN,EXISTS,NOT EXISTS的用法和差別SQL
- MYSQL和ORACLE時區設定比較MySqlOracle
- SAP ERP 與 Oracle ERP 比較(轉)Oracle
- oracle中字串的大小比較,字串與數字的比較和運算Oracle字串
- Oracle 中不使用NOT IN 和 NOT EXISTS的另一種方法Oracle
- Java 中 Comparable 和 Comparator 比較Java
- Go中泛型和反射比較指南Go泛型反射
- 堆和棧在程式中的比較
- powershell中的where和foreach比較
- oracle 比較日期相等Oracle
- oracle sql日期比較:OracleSQL