資源描述:
《ORACLE基于SQL查詢優(yōu)化培訓(xùn)文檔.ppt》由會(huì)員上傳分享,免費(fèi)在線閱讀,更多相關(guān)內(nèi)容在教育資源-天天文庫(kù)。
1、ORACLE基于SQL查詢優(yōu)化培訓(xùn)文檔2013年11月查詢優(yōu)化基本準(zhǔn)則詳解2ORACLE的解析器按照從右到左的順序處理FROM子句中的表名,因此FROM子句中寫(xiě)在最后的表(基礎(chǔ)表drivingtable)將被最先處理。在FROM子句中包含多個(gè)表的情況下,你必須選擇記錄條數(shù)最少的表作為基礎(chǔ)表。例如:表TAB116384條記錄表TAB21條記錄選擇TAB2作為基礎(chǔ)表(最好的方法)selectcount(*)fromtab1,tab2執(zhí)行時(shí)間0.96秒選擇TAB2作為基礎(chǔ)表(不佳的方法)selectcount(*)fromtab2,tab1執(zhí)行時(shí)間26.09秒查詢優(yōu)
2、化基本準(zhǔn)則詳解3ORACLE采用自下而上的順序解析WHERE子句,根據(jù)這個(gè)原理,表之間的連接必須寫(xiě)在其他WHERE條件之前,那些可以過(guò)濾掉最大數(shù)量記錄的條件必須寫(xiě)在WHERE子句的末尾。例如:(低效,執(zhí)行時(shí)間156.3秒)SELECT…FROMEMPEWHERESAL>50000ANDJOB=‘MANAGER'AND25<(SELECTCOUNT(*)FROMEMPWHEREMGR=E.EMPNO);(高效,執(zhí)行時(shí)間10.6秒)SELECT…FROMEMPEWHERE25<(SELECTCOUNT(*)FROMEMPWHEREMGR=E.EMPNO)ANDSA
3、L>50000ANDJOB=‘MANAGER';在進(jìn)行多表關(guān)聯(lián)時(shí),多用Where語(yǔ)句把單個(gè)表的結(jié)果集最小化,多用聚合函數(shù)匯總結(jié)果集后再與其它表做關(guān)聯(lián),以使結(jié)果集數(shù)據(jù)量最小化查詢優(yōu)化基本準(zhǔn)則詳解4減少對(duì)表的查詢?cè)诤凶硬樵兊腟QL語(yǔ)句中,要特別注意減少對(duì)表的查詢例如:低效SELECTTAB_NAMEFROMTABLESWHERETAB_NAME=(SELECTTAB_NAMEFROMTAB_COLUMNSWHEREVERSION=604)ANDDB_VER=(SELECTDB_VERFROMTAB_COLUMNSWHEREVERSION=604)高效SELECT
4、TAB_NAMEFROMTABLESWHERE(TAB_NAME,DB_VER)=(SELECTTAB_NAME,DB_VER)FROMTAB_COLUMNSWHEREVERSION=604)查詢優(yōu)化基本準(zhǔn)則詳解5用EXISTS替代IN在許多基于基礎(chǔ)表的查詢中,為了滿足一個(gè)條件,往往需要對(duì)另一個(gè)表進(jìn)行聯(lián)接。在這種情況下,使用EXISTS(或NOTEXISTS)通常將提高查詢的效率。使用exists而不用IN因?yàn)镋xists只檢查行的存在,而in檢查實(shí)際值例如:低效SELECT*FROMEMP(基礎(chǔ)表)WHEREEMPNO>0ANDDEPTNOIN(SELECT
5、DEPTNOFROMDEPTWHERELOC=‘MELB')高效SELECT*FROMEMP(基礎(chǔ)表)WHEREEMPNO>0ANDEXISTS(SELECT‘X'FROMDEPTWHEREDEPT.DEPTNO=EMP.DEPTNOANDLOC=‘MELB')用IN的SQL性能總是比較低,原因是:對(duì)于用IN的SQL語(yǔ)句ORACLE總是試圖將其轉(zhuǎn)換成多個(gè)表的連接,如果轉(zhuǎn)換不成功則先執(zhí)行IN里面的子查詢,再查詢外層的表記錄如果轉(zhuǎn)換成功就轉(zhuǎn)換成多個(gè)表的連接。因此不管理怎么,用IN的SQL語(yǔ)句總是多了一個(gè)轉(zhuǎn)換的過(guò)程。因此在業(yè)務(wù)密集的SQL當(dāng)中盡量不采用IN操作符。查
6、詢優(yōu)化基本準(zhǔn)則詳解6用NOTEXISTS替代NOTIN在子查詢中,NOTIN子句將執(zhí)行一個(gè)內(nèi)部的排序和合并。無(wú)論在哪種情況下,NOTIN都是最低效的(因?yàn)樗鼘?duì)子查詢中的表執(zhí)行了一個(gè)全表遍歷)。為了避免使用NOTIN,我們可以把它改寫(xiě)成外連接(OuterJoins)或NOTEXISTS.例如:SELECT…FROMEMPWHEREDEPT_NONOTIN(SELECTDEPT_NOFROMDEPTWHEREDEPT_CAT='A');為了提高效率。改寫(xiě)為:SELECT….FROMEMPEWHERENOTEXISTS(SELECT‘X'FROMDEPTDWHERE
7、D.DEPT_NO=E.DEPT_NOANDDEPT_CAT=‘A');NOTIN操作符此操作強(qiáng)烈推薦不使用,因?yàn)槠洳荒軕?yīng)用表的索引。如遇這種情況,應(yīng)該用EXISTS,NOTEXISTS方案代替。查詢優(yōu)化基本準(zhǔn)則詳解7用EXISTS替換DISTINCT當(dāng)提交一個(gè)包含一對(duì)多表信息(比如部門(mén)表和雇員表)的查詢時(shí),避免在SELECT子句中使用DISTINCT.一般可以考慮用EXIST替換例如:低效:SELECTDISTINCTDEPT_NO,DEPT_NFROMDEPTD,EMPEWHERED.DEPT_NO=E.DEPT_NO高效:SELECTDEPT_NO,DE
8、PT_NAMEFROMDEPTDWHE