ORACLESQL性能優(yōu)化系列(十三)

字號(hào):

43. 用WHERE替代ORDER BY
    ORDER BY 子句只在兩種嚴(yán)格的條件下使用索引.
    ORDER BY中所有的列必須包含在相同的索引中并保持在索引中的排列順序.
    ORDER BY中所有的列必須定義為非空.
    WHERE子句使用的索引和ORDER BY子句中所使用的索引不能并列.
    例如:
     表DEPT包含以下列:
     :smarttags" />DEPT_CODE PK NOT NULL
     DEPT_DESC NOT NULL
     DEPT_TYPE NULL
     非性的索引(DEPT_TYPE)
     低效: (索引不被使用)
     SELECT DEPT_CODE
     FROM DEPT
     ORDER BY DEPT_TYPE
     EXPLAIN PLAN:
     SORT ORDER BY
     TABLE ACCESS FULL
     高效: (使用索引)
     SELECT DEPT_CODE
     FROM DEPT
     WHERE DEPT_TYPE > 0
     EXPLAIN PLAN:
     TABLE ACCESS BY ROWID ON EMP
     INDEX RANGE SCAN ON DEPT_IDX
    譯者按:
     ORDER BY 也能使用索引! 這的確是個(gè)容易被忽視的知識(shí)點(diǎn). 我們來(lái)驗(yàn)證一下:
    SQL> select * from emp order by empno;
    Execution Plan
    ----------------------------------------------------------
     0 SELECT STATEMENT Optimizer=CHOOSE
     1 0 TABLE ACCESS (BY INDEX ROWID) OF ’EMP’
     2 1 INDEX (FULL SCAN) OF ’EMPNO’ (UNIQUE)
    44. 避免改變索引列的類型.
    當(dāng)比較不同數(shù)據(jù)類型的數(shù)據(jù)時(shí), ORACLE自動(dòng)對(duì)列進(jìn)行簡(jiǎn)單的類型轉(zhuǎn)換.
    假設(shè) EMPNO是一個(gè)數(shù)值類型的索引列.
    SELECT …
    FROM EMP
    WHERE EMPNO = ‘123’
    實(shí)際上,經(jīng)過(guò)ORACLE類型轉(zhuǎn)換, 語(yǔ)句轉(zhuǎn)化為:
    SELECT …
    FROM EMP
    WHERE EMPNO = TO_NUMBER(‘123’)
    幸運(yùn)的是,類型轉(zhuǎn)換沒(méi)有發(fā)生在索引列上,索引的用途沒(méi)有被改變.
    現(xiàn)在,假設(shè)EMP_TYPE是一個(gè)字符類型的索引列.
    SELECT …
    FROM EMP
    WHERE EMP_TYPE = 123
    這個(gè)語(yǔ)句被ORACLE轉(zhuǎn)換為:
    SELECT …
    FROM EMP
    WHERE TO_NUMBER(EMP_TYPE)=123
    因?yàn)閮?nèi)部發(fā)生的類型轉(zhuǎn)換, 這個(gè)索引將不會(huì)被用到!
    譯者按:
    為了避免ORACLE對(duì)你的SQL進(jìn)行隱式的類型轉(zhuǎn)換, 把類型轉(zhuǎn)換用顯式表現(xiàn)出來(lái). 注意當(dāng)字符和數(shù)值比較時(shí), ORACLE會(huì)優(yōu)先轉(zhuǎn)換數(shù)值類型到字符類型.
    45. 需要當(dāng)心的WHERE子句
    某些SELECT 語(yǔ)句中的WHERE子句不使用索引. 這里有一些例子.
    在下面的例子里, ‘!=’ 將不使用索引. 記住, 索引只能告訴你什么存在于表中, 而不能告訴你什么不存在于表中.
    不使用索引:
    SELECT ACCOUNT_NAME
    FROM TRANSACTION
    WHERE AMOUNT !=0;