** 36. ** ** 用 UNION ** ** 替换 OR ( ** ** 适用于索引列 ) **
通常情况下 , 用 UNION 替换 WHERE 子句中的 OR 将会起到较好的效果 . 对索引列使用 OR 将造成全表扫描 . 注意 , 以上规则只针对多个索引列有效 . 如果有 column 没有被索引 , 查询效率可能会因为你没有选择 OR 而降低 .
在下面的例子中 , LOC_ID 和 REGION上 都建有索引 .
高效 :
SELECT LOC_ID , LOC_DESC , REGION
FROM LOCATION
WHERE LOC_ID = 10
UNION
SELECT LOC_ID , LOC_DESC , REGION
FROM LOCATION
WHERE REGION = “ MELBOURNE ”
低效 :
SELECT LOC_ID , LOC_DESC , REGION
FROM LOCATION
WHERE LOC_ID = 10 OR REGION = “ MELBOURNE ”
如果你坚持要用 OR, 那就需要返回记录最少的索引列写在最前面 .
注意 :
WHERE KEY1 = 10 ( 返回最少记录 )
OR KEY2 = 20 ( 返回最多记录 )
ORACLE 内部将以上转换为
WHERE KEY1 = 10 AND
((NOT KEY1 = 10) AND KEY2 = 20)
** 译者按 : **
** 下面的测试数据仅供参考 : (a = 1003 ** ** 返回一条记录 , b = 1 ** ** 返回 1003 ** ** 条记录 ) **
** SQL> select * from unionvsor /1st test/ **
** 2 where a = 1003 or b = 1; **
** 1003 rows selected. **
** Execution Plan **
** ---------------------------------------------------------- **
** 0 SELECT STATEMENT Optimizer=CHOOSE **
** 1 0 CONCATENATION **
** 2 1 TABLE ACCESS (BY INDEX ROWID) OF 'UNIONVSOR' **
** 3 2 INDEX (RANGE SCAN) OF 'UB' (NON-UNIQUE) **
** 4 1 TABLE ACCESS (BY INDEX ROWID) OF 'UNIONVSOR' **
** 5 4 INDEX (RANGE SCAN) OF 'UA' (NON-UNIQUE) **
** Statistics **
** ---------------------------------------------------------- **
** 0 recursive calls **
** 0 db block gets **
** 144 consistent gets **
** 0 physical reads **
** 0 redo size **
** 63749 bytes sent via SQL*Net to client **
** 7751 bytes received via SQL*Net from client **
** 68 SQL*Net roundtrips to/from client **
** 0 sorts (memory) **
** 0 sorts (disk) **
** 1003 rows processed **
** SQL> select * from unionvsor /2nd test/ **
** 2 where b = 1 or a = 1003 ; **
** 1003 rows selected. **
** Execution Plan **
** ---------------------------------------------------------- **
** 0 SELECT STATEMENT Optimizer=CHOOSE **
** 1 0 CONCATENATION **
** 2 1 TABLE ACCESS (BY INDEX ROWID) OF 'UNIONVSOR' **
** 3 2 INDEX (RANGE SCAN) OF 'UA' (NON-UNIQUE) **
** 4 1 TABLE ACCESS (BY INDEX ROWID) OF 'UNIONVSOR' **
** 5 4 INDEX (RANGE SCAN) OF 'UB' (NON-UNIQUE) **
** Statistics **
** ---------------------------------------------------------- **
** 0 recursive calls **
** 0 db block gets **
** 143 consistent gets **
** 0 physical reads **
** 0 redo size **
** 63749 bytes sent via SQL*Net to client **
** 7751 bytes received via SQL*Net from client **
** 68 SQL*Net roundtrips to/from client **
** 0 sorts (memory) **
** 0 sorts (disk) **
** 1003 rows processed **
** SQL> select * from unionvsor /3rd test/ **
** 2 where a = 1003 **
** 3 union **
** 4 select * from unionvsor **
** 5 where b = 1; **
** 1003 rows selected. **
** Execution Plan **
** ---------------------------------------------------------- **
** 0 SELECT STATEMENT Optimizer=CHOOSE **
** 1 0 SORT (UNIQUE) **
** 2 1 UNION-ALL **
** 3 2 TABLE ACCESS (BY INDEX ROWID) OF 'UNIONVSOR' **
** 4 3 INDEX (RANGE SCAN) OF 'UA' (NON-UNIQUE) **
** 5 2 TABLE ACCESS (BY INDEX ROWID) OF 'UNIONVSOR' **
** 6 5 INDEX (RANGE SCAN) OF 'UB' (NON-UNIQUE) **
** Statistics **
** ---------------------------------------------------------- **
** 0 recursive calls **
** 0 db block gets **
** 10 consistent gets **
** 0 physical reads **
** 0 redo size **
** 63735 bytes sent via SQL*Net to client **
** 7751 bytes received via SQL*Net from client **
** 68 SQL*Net roundtrips to/from client **
** 1 sorts (memory) **
** 0 sorts (disk) **
** 1003 rows processed **
** 用 UNION ** ** 的效果可以从 consistent gets ** ** 和 SQL*NET ** ** 的数据交换量的减少看出 **
** 37. ** ** 用 IN ** ** 来替换 OR **
下面的查询可以被更有效率的语句替换 :
低效 :
SELECT….
FROM LOCATION
WHERE LOC_ID = 10
OR LOC_ID = 20
OR LOC_ID = 30
高效
SELECT…
FROM LOCATION
WHERE LOC_IN IN (10,20,30);
** 译者按 : **
** 这是一条简单易记的规则,但是实际的执行效果还须检验,在 ORACLE8i ** ** 下,两者的执行路径似乎是相同的. **
** 38. ** ** 避免在索引列上使用 IS NULL ** ** 和 IS NOT NULL **
避免在索引中使用任何可以为空的列, ORACLE 将无法使用该索引 .对于单列索引,如果列包含空值,索引中将不存在此记录 . 对于复合索引,如果每个列都为空,索引中同样不存在此记录 . 如果至少有一个列不为空,则记录存在于索引中.
举例 :
如果唯一性索引建立在表的 A 列和 B 列上 , 并且表中存在一条记录的 A,B 值为 (123,null) , ORACLE 将不接受下一条具有相同 A,B 值( 123,null )的记录 ( 插入 ). 然而如果
所有的索引列都为空, ORACLE 将认为整个键值为空而 空不等于空 . 因此你可以插入 1000
条具有相同键值的记录 , 当然它们都是空!
因为空值不存在于索引列中 , 所以 WHERE 子句中对索引列进行空值比较将使 ORACLE 停用该索引 .
举例 :
低效 : ( 索引失效 )
SELECT …
FROM DEPARTMENT
WHERE DEPT_CODE **IS NOT NULL; **
高效 : ( 索引有效 )
SELECT …
FROM DEPARTMENT
WHERE DEPT_CODE ** >=0; **