在调查一个性能问题的时候,一个同事问道,为什么数据库有些时候这么不聪明,明明表上有索引,但是在执行一个简单的count的时候居然全表扫描了!难道不知道走索引更快么?
试图从最简单的count来重新了解oracle查询计划的选择,以及最终产生的结果。虽然有些结果会让人觉得有些意外,并且可能会鄙视,这个查询 计划选择真的不够聪明。但稍微用心点的去了解,做的已经足够细致了。大多数情况下,根据我们输入的信息,来自输入的SQL、表结构、索引状况、统计信息, 会得出一个比较优的计划。所以和前面一直试图讲到索引和join方式一样,所有这样的选择不是因为数据库厂商这样规定的,而是基于存储的数据的实际情况, 就应该(甚至说不得不)这么去选择。
-- Create table createtableIDOUBA.APP_APPLICATIONAUDIT ( ID NUMBER, UNIQUEID NUMBER, POLICYID NUMBER, IP VARCHAR2(200)notnull, IPNUM NUMBERnotnull, MAC VARCHAR2(200), USERVISIT VARCHAR2(100), PHONE VARCHAR2(200), GROUPNAME VARCHAR2(100), PORT NUMBER, APP_URL VARCHAR2(200), TITLE VARCHAR2(200), REQUESTS VARCHAR2(1000), REQIDENTITYCARDVARCHAR2(1000), REQKEY VARCHAR2(1000), RESPONSEEKEY VARCHAR2(3000), UPDATETIME DATE, AUDITTYPE NUMBER, TITLEID NUMBER, SUBTITLEID NUMBER, SERVERIP VARCHAR2(200), DOMAINNAME VARCHAR2(200) ) tablespaceUSERS pctfree10 initrans1 maxtrans255 storage ( initial64K minextents1 maxextentsunlimited ); -- Create/Recreate indexes createindexINDEX_UPDATETIMEonIDOUBA.APP_APPLICATIONAUDIT(UPDATETIME) tablespaceSYSTEM pctfree10 initrans2 maxtrans255 storage ( initial64K minextents1 maxextentsunlimited );
表中有20001000条记录,有二十多个字段,包括几个长度比较大的字段。为了实验测试,只是讲IP字段设置为not null。在UpdateTime上建了默认的B树索引。
分为三种场景来讨论,然后比较其结果,然后分析该结果。说明数据库的执行计划的选择其实是足够聪明的。
selectcount(*)fromIDOUBA.App_ApplicationAudit; --count(*) selectcount(PolicyId)fromIDOUBA.App_ApplicationAudit; --count(可以null的列) selectcount(IP)fromIDOUBA.App_ApplicationAudit; --count(不可null的列) selectcount(UpdateTime)fromIDOUBA.App_ApplicationAudit; --count(可以为null并且建了索引的列)
场景二:
selectcount(*)fromIDOUBA.App_ApplicationAuditwhereupdatetime>to_date(‘2013-05-01 10:23:44‘,‘yyyy-mm-dd hh24:mi:ss‘); --count(*) selectcount(PolicyId)fromIDOUBA.App_ApplicationAuditwhereupdatetime>to_date(‘2013-05-01 10:23:44‘,‘yyyy-mm-dd hh24:mi:ss‘); --count(可以null的列) selectcount(IP)fromIDOUBA.App_ApplicationAuditwhereupdatetime>to_date(‘2013-05-01 10:23:44‘,‘yyyy-mm-dd hh24:mi:ss‘); --count(不可null的列) selectcount(UpdateTime)fromyIDOUBA.App_ApplicationAuditwhereupdatetime>to_date(‘2013-05-01 10:23:44‘,‘yyyy-mm-dd hh24:mi:ss‘); --count(可以为null并且建了索引的列)
场景三:
selectcount(*)fromIDOUBA.App_ApplicationAudit; --count(*) selectcount(PolicyId)fromIDOUBA.App_ApplicationAudit; --count(可以null的列) selectcount(IP)fromIDOUBA.App_ApplicationAudit; --count(不可null的列) selectcount(UpdateTime)fromIDOUBA.App_ApplicationAudit; --count(可以为null并且建了索引的列)
场景/Count类型 | Count(*) | Count(PolicyId) 非索引可null的列 | Count(IP)非索引not null 的列 | Count(Updatetime)可以null的索引列 | Count(IPNum) not null 的索引列 | ||||||||||
执行时间(秒) | 物理读 | 执行操作 | 执行时间(秒) | 物理读 | 执行操作 | 执行时间(秒) | 物理读 | 执行操作 | 执行时间(秒) | 物理读 | 执行操作 | 执行时间(秒) | 物理读 physical reads | 执行操作 | |
场景一 | 50.98 | 51671 | 表扫描 | 47.43 | 451672 | 表扫描 | 47.73 | 451672 | 表扫描 | 8.7 | 82393 | UpdateTime上快速索引扫描 | 无 | 无 | 无 |
场景二 | 9.28 | 82395 | UpdateTime上快速索引扫描 | 48.12 | 451673 | 表扫描 | 9.65 | 82428 | UpdateTime上快速索引扫描 | 9.03 | 82394 | UpdateTime上快速索引扫描 | 无 | 无 | 无 |
场景三 | 5.14 | 50224 | IpNum上快速索引扫描 | 47.89 | 451672 | 表扫描 | 2.25 | 15153 | IpNum上快速索引扫描 | 9.06 | 82393 | UpdateTime上快速索引扫描 | 3.2 | 82393 | IpNum上快速索引扫描 |
其实观察查询计划,也就两种,一种快速索引扫描(INDEX FAST FULL SCAN),另外一种是全表扫描(TABLE ACCESS FULL|)。因为实验表中数据量还是不算小,有两千多万,两者的对比还是比较明显。主要体现在总的执行时间,和物理读取上面,差着一个量级。下面对表中 的执行结果逐个进行总结和解释。
当表中只在一个可以null的列上建索引的情况下。观察到count(*) 、count not null的列count(IP),count 可以null的列PolicyId的count(PolicyId)的效果一样都是走全表扫描(TABLE ACCESS FULL|);而对于count建了索引并且列可以null的列count(UpdateTime),走快速索引扫描(INDEX FAST FULL SCAN)。即只在索引列UpdateTime上, count会走快速索引扫描(INDEX FAST FULL SCAN),其他都是全表扫描。
解释:因为oracle不会索引null,因此扫描该列上的索引只能知道该列上非null的值,而count(*) 和count(非空列)都是统计总行数,从索引上不能获得该信息,只能全表扫描。而count(可空列)就更不能参照该索引了,建了索引的可空列上非 null的行数,和在count的可空列上非null的行数没有任何关系,因此也只能全表扫描了。
比场景一where子句中包含了可以null的列update的条件where updatetime > to_date(’2013-05-01 10:23:44′, ‘yyyy-mm-dd hh24:mi:ss’) 。观察到count(*)和count not null的列count(IP)和count UpdateTime一样都走updatetime上的快速索引扫描(INDEX FAST FULL SCAN);而count 可null列PolicyId的count(PolicyId)还是全表扫描。
解释:满足类似于pdatetime > to_date(’2013-05-01 10:23:44′, ‘yyyy-mm-dd hh24:mi:ss’)这样可空列上的条件,其实隐含的意思是满足该条件并且该列上取值不为null的行的行数。则在这列上的 count,count(*),在其他非null列上的count,表达的都是这个意思,因此可以利用该索引来做索引扫描。而在另外一个可null的列上 的count是表示满足该where条件同时在count列上值非null的记录行数,索引列上不为空的行可能在该列上为空,因此不能参照那个索引,只能 全表扫描。
当表中存在一个not null的列IpNum上建了索引。观察到count(*)和count(IP)在非空索引上快速索引扫描(INDEX FAST FULL SCAN),但是count(ProjectId)还是走原来的全表扫描。即:当表上在一个not null列上建了索引,则只有可Null的列的count走全表扫描,其他的都会走这个建了索引的no null 列的快速索引扫描(INDEX FAST FULL SCAN)。
解释:有一个not null的列上建了索引,则这个索引上的记录数就是表的行数,count(*),count(非空列)都是数行数。但是count(可null 列)是数这个列上不为空的记录数,因此不能参照索引,只能全表扫描了。
在Oracle中,Count(Column)是计算Column上不为该列取值不为null的行数,Count一个not null的列其实就是总行数。而Count(*)是不区分null或者not null就是所有记录行数。因此二者语义是一样的,实验也证明了在各种情况下其执行计划总是一样的。对于这两种Count,表上无论哪个索引能提供这样的 语义(在例子中场景二和场景三的两个不同索引分别提供了这样的语义),就会走这个索引。如果没有索引能提供这个语义,就不得不走全表扫描了。而Count 一个可以为null的列,因为要数本列上到底有多少行的值不为null,因此不能参照别的列,必须在该列上数数,如果该列上有索引,则会在该列的索引上扫 描,如果该列上没有索引,则不得不全表扫描。
1. 只是在有一个可以为null的列UpdateTime上建了一个索引的Count(*)
SQL>selectcount(*)fromIDOUBA.App_ApplicationAudit; COUNT(*) ---------- 20001000 已用时间: 00:00:50.98 执行计划 ---------------------------------------------------------- Planhashvalue:2649150711 ----------------------------------------------------------------------------------- |Id |Operation |Name |Rows |Cost(%CPU)|Time | ----------------------------------------------------------------------------------- | 0|SELECTSTATEMENT | | 1|99985 (2)|00:20:00| | 1| SORTAGGREGATE | | 1| | | | 2| TABLEACCESSFULL|APP_APPLICATIONAUDIT| 19M|99985 (2)|00:20:00| ----------------------------------------------------------------------------------- 统计信息 ---------------------------------------------------------- 166 recursivecalls 0 dbblockgets 451721 consistentgets 451671 physicalreads 0 redosize 410 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 5 sorts(memory) 0 sorts(disk) 1 rowsprocessed
2. 只是在有一个可以为null的列UpdateTime上建了一个索引的Count一个可以null的列policyId
SQL>selectcount(policyId)fromIDOUBA.App_ApplicationAudit; COUNT(POLICYID) --------------- 20001000 已用时间: 00:00:47.43 执行计划 ---------------------------------------------------------- Planhashvalue:2649150711 ------------------------------------------------------------------------------------------- |Id |Operation |Name |Rows |Bytes|Cost(%CPU)|Time | ------------------------------------------------------------------------------------------- | 0|SELECTSTATEMENT | | 1| 2| 100K (2)|00:20:02| | 1| SORTAGGREGATE | | 1| 2| | | | 2| TABLEACCESSFULL|APP_APPLICATIONAUDIT| 19M| 38M| 100K (2)|00:20:02| ------------------------------------------------------------------------------------------- 统计信息 ---------------------------------------------------------- 1 recursivecalls 0 dbblockgets 451701 consistentgets 451672 physicalreads 0 redosize 417 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 0 sorts(memory) 0 sorts(disk) 1 rowsprocessed
3. 只是在有一个可以为null的列UpdateTime上建了一个索引的Count一个可以null的索引列updatetime:Count(UpdateTime)
SQL>selectcount(updatetime)fromIDOUBA.App_ApplicationAudit; COUNT(UPDATETIME) ----------------- 20001000 已用时间: 00:00:08.70 执行计划 ---------------------------------------------------------- Planhashvalue:3909306724 ------------------------------------------------------------------------------------------ |Id |Operation |Name |Rows |Bytes|Cost(%CPU)|Time | ------------------------------------------------------------------------------------------ | 0|SELECTSTATEMENT | | 1| 8|18204 (3)|00:03:39| | 1| SORTAGGREGATE | | 1| 8| | | | 2| INDEXFASTFULLSCAN|INDEX_UPDATETIME| 19M| 152M|18204 (3)|00:03:39| ------------------------------------------------------------------------------------------ 统计信息 ---------------------------------------------------------- 164 recursivecalls 0 dbblockgets 82448 consistentgets 82393 physicalreads 0 redosize 419 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 5 sorts(memory) 0 sorts(disk) 1 rowsprocessed
4. 只是在有一个可以为null的列UpdateTime上建了一个索引的Count一个not null的非索引列IP:Count(IP)
SQL>selectcount(Ip)fromIDOUBA.App_ApplicationAudit; COUNT(IP) ---------- 20001000 已用时间: 00:00:47.73 执行计划 ---------------------------------------------------------- Planhashvalue:2649150711 ----------------------------------------------------------------------------------- |Id |Operation |Name |Rows |Cost(%CPU)|Time | ----------------------------------------------------------------------------------- | 0|SELECTSTATEMENT | | 1|99985 (2)|00:20:00| | 1| SORTAGGREGATE | | 1| | | | 2| TABLEACCESSFULL|APP_APPLICATIONAUDIT| 19M|99985 (2)|00:20:00| ----------------------------------------------------------------------------------- 统计信息 ---------------------------------------------------------- 218 recursivecalls 0 dbblockgets 451730 consistentgets 451672 physicalreads 0 redosize 411 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 5 sorts(memory) 0 sorts(disk) 1 rowsprocessed
5. 只是在有一个可以为null的列UpdateTime上建了一个索引,Query的Where子句中包含该UpdateTime列的条件where updatetime > to_date(’2013-05-01 10:23:44′, ‘yyyy-mm-dd hh24:mi:ss’) ,Count(*)
SQL>selectcount(*)fromIDOUBA.App_ApplicationAudit whereupdatetime>to_date(‘2013-05-01 10:23:44‘,‘yyyy-mm-dd hh24:mi:ss‘); COUNT(*) ---------- 20001000 已用时间: 00:00:09.28 执行计划 ---------------------------------------------------------- Planhashvalue:3909306724 ------------------------------------------------------------------------------------------ |Id |Operation |Name |Rows |Bytes|Cost(%CPU)|Time | ------------------------------------------------------------------------------------------ | 0|SELECTSTATEMENT | | 1| 8|18373 (4)|00:03:41| | 1| SORTAGGREGATE | | 1| 8| | | |* 2| INDEXFASTFULLSCAN|INDEX_UPDATETIME| 19M| 152M|18373 (4)|00:03:41| ------------------------------------------------------------------------------------------ PredicateInformation(identifiedbyoperationid): --------------------------------------------------- 2-filter("UPDATETIME">TO_DATE(‘2013-05-01 10:23:44‘,‘yyyy-mm-dd hh24:mi:ss‘)) 统计信息 ---------------------------------------------------------- 227 recursivecalls 0 dbblockgets 82464 consistentgets 82395 physicalreads 0 redosize 410 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 5 sorts(memory) 0 sorts(disk) 1 rowsprocessed
6. 只是在有一个可以为null的列UpdateTime上建了一个索引,Query的Where子句中包含该UpdateTime列的条件where updatetime > to_date(’2013-05-01 10:23:44′, ‘yyyy-mm-dd hh24:mi:ss’) 。Count一个可以null的列PolicyId:Count(PolicyId)
SQL>selectcount(policyId)fromIDOUBA.App_ApplicationAudit whereupdatetime>to_date(‘2013-05-01 10:23:44‘,‘yyyy-mm-dd hh24:mi:ss‘); COUNT(POLICYID) --------------- 20001000 已用时间: 00:00:48.12 执行计划 ---------------------------------------------------------- Planhashvalue:2649150711 ------------------------------------------------------------------------------------------- |Id |Operation |Name |Rows |Bytes|Cost(%CPU)|Time | ------------------------------------------------------------------------------------------- | 0|SELECTSTATEMENT | | 1| 10| 101K (3)|00:20:20| | 1| SORTAGGREGATE | | 1| 10| | | |* 2| TABLEACCESSFULL|APP_APPLICATIONAUDIT| 19M| 190M| 101K (3)|00:20:20| ------------------------------------------------------------------------------------------- PredicateInformation(identifiedbyoperationid): --------------------------------------------------- 2-filter("UPDATETIME">TO_DATE(‘2013-05-01 10:23:44‘,‘yyyy-mm-dd hh24:mi:ss‘)) 统计信息 ---------------------------------------------------------- 227 recursivecalls 0 dbblockgets 451737 consistentgets 451673 physicalreads 0 redosize 417 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 5 sorts(memory) 0 sorts(disk) 1 rowsprocessed
7. 只是在有一个可以为null的列UpdateTime上建了一个索引,Query的Where子句中包含该UpdateTime列的条件where updatetime > to_date(’2013-05-01 10:23:44′, ‘yyyy-mm-dd hh24:mi:ss’) 。Count一个not null的列IP:Count(IP)
SQL>selectcount(ip)fromIDOUBA.App_ApplicationAudit whereupdatetime>to_date(‘2013-05-01 10:23:44‘,‘yyyy-mm-dd hh24:mi:ss‘); COUNT(IP) ---------- 20001000 已用时间: 00:00:09.65 执行计划 ---------------------------------------------------------- Planhashvalue:3909306724 ------------------------------------------------------------------------------------------ |Id |Operation |Name |Rows |Bytes|Cost(%CPU)|Time | ------------------------------------------------------------------------------------------ | 0|SELECTSTATEMENT | | 1| 8|18373 (4)|00:03:41| | 1| SORTAGGREGATE | | 1| 8| | | |* 2| INDEXFASTFULLSCAN|INDEX_UPDATETIME| 19M| 152M|18373 (4)|00:03:41| ------------------------------------------------------------------------------------------ PredicateInformation(identifiedbyoperationid): --------------------------------------------------- 2-filter("UPDATETIME">TO_DATE(‘2013-05-01 10:23:44‘,‘yyyy-mm-dd hh24:mi:ss‘)) 统计信息 ---------------------------------------------------------- 1 recursivecalls 0 dbblockgets 82428 consistentgets 82393 physicalreads 0 redosize 411 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 0 sorts(memory) 0 sorts(disk) 1 rowsprocessed
8. 只是在有一个可以为null的列UpdateTime上建了一个索引,Query的Where子句中包含该UpdateTime列的条件where updatetime > to_date(’2013-05-01 10:23:44′, ‘yyyy-mm-dd hh24:mi:ss’) 。Count该可以 null的索引列UpdateTime:Count(UpdateTime)
SQL>selectcount(updatetime)fromIDOUBA.App_ApplicationAuditwhereupdatetime>to_date(‘2013-05-01 10:23:44‘,‘yyyy-mm-dd hh24:mi:ss‘); COUNT(UPDATETIME) ----------------- 20001000 已用时间: 00:00:09.03 执行计划 ---------------------------------------------------------- Planhashvalue:3909306724 ------------------------------------------------------------------------------------------ |Id |Operation |Name |Rows |Bytes|Cost(%CPU)|Time | ------------------------------------------------------------------------------------------ | 0|SELECTSTATEMENT | | 1| 8|18204 (3)|00:03:39| | 1| SORTAGGREGATE | | 1| 8| | | | 2| INDEXFASTFULLSCAN|INDEX_UPDATETIME| 19M| 152M|18204 (3)|00:03:39| ------------------------------------------------------------------------------------------ 统计信息 ---------------------------------------------------------- 185 recursivecalls 0 dbblockgets 82460 consistentgets 82394 physicalreads 0 redosize 419 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 5 sorts(memory) 0 sorts(disk) 1 rowsprocessed
9. 在一个not null的列IPNum创建索引count(*)
SQL>selectcount(*)fromIDOUBA.App_ApplicationAudit; COUNT(*) ---------- 20001000 已用时间: 00:00:05.14 执行计划 ---------------------------------------------------------- Planhashvalue:2581539037 ----------------------------------------------------------------------------- |Id |Operation |Name |Rows |Cost(%CPU)|Time | ----------------------------------------------------------------------------- | 0|SELECTSTATEMENT | | 1|11435 (5)|00:02:18| | 1| SORTAGGREGATE | | 1| | | | 2| INDEXFASTFULLSCAN|INDEX_IPNUM| 19M|11435 (5)|00:02:18| ----------------------------------------------------------------------------- 统计信息 ---------------------------------------------------------- 1 recursivecalls 0 dbblockgets 50253 consistentgets 50224 physicalreads 0 redosize 410 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 0 sorts(memory) 0 sorts(disk) 1 rowsprocessed
10 在一个not null的列IPNum创建索引。Count可null的列PolicyId:Count(PolicyId)
SQL>selectcount(policyId)fromIDOUBA.App_ApplicationAudit; COUNT(POLICYID) --------------- 20001000 已用时间: 00:00:47.89 执行计划 ---------------------------------------------------------- Planhashvalue:2649150711 ------------------------------------------------------------------------------------------- |Id |Operation |Name |Rows |Bytes|Cost(%CPU)|Time | ------------------------------------------------------------------------------------------- | 0|SELECTSTATEMENT | | 1| 2| 100K (2)|00:20:02| | 1| SORTAGGREGATE | | 1| 2| | | | 2| TABLEACCESSFULL|APP_APPLICATIONAUDIT| 19M| 38M| 100K (2)|00:20:02| ------------------------------------------------------------------------------------------- 统计信息 ---------------------------------------------------------- 1 recursivecalls 0 dbblockgets 451701 consistentgets 451672 physicalreads 0 redosize 417 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 0 sorts(memory) 0 sorts(disk) 1 rowsprocessed
11 在一个not null的列IPNum创建索引。Count not null的列IP:Count(IP)
SQL>selectcount(Ip)fromIDOUBA.App_ApplicationAudit; COUNT(IP) ---------- 20001000 已用时间: 00:00:02.25 执行计划 ---------------------------------------------------------- Planhashvalue:2581539037 ----------------------------------------------------------------------------- |Id |Operation |Name |Rows |Cost(%CPU)|Time | ----------------------------------------------------------------------------- | 0|SELECTSTATEMENT | | 1|11435 (5)|00:02:18| | 1| SORTAGGREGATE | | 1| | | | 2| INDEXFASTFULLSCAN|INDEX_IPNUM| 19M|11435 (5)|00:02:18| ----------------------------------------------------------------------------- 统计信息 ---------------------------------------------------------- 196 recursivecalls 0 dbblockgets 50290 consistentgets 15153 physicalreads 0 redosize 411 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 5 sorts(memory) 0 sorts(disk) 1 rowsprocessed
12 在一个not null的列IPNum创建索引。Count另外一个建了索引的可null的列UpdateTime:Count(UpdateTime)
SQL>selectcount(updatetime)fromIDOUBA.App_ApplicationAudit; COUNT(UPDATETIME) ----------------- 20001000 已用时间: 00:00:09.06 执行计划 ---------------------------------------------------------- Planhashvalue:3909306724 ------------------------------------------------------------------------------------------ |Id |Operation |Name |Rows |Bytes|Cost(%CPU)|Time | ------------------------------------------------------------------------------------------ | 0|SELECTSTATEMENT | | 1| 8|18204 (3)|00:03:39| | 1| SORTAGGREGATE | | 1| 8| | | | 2| INDEXFASTFULLSCAN|INDEX_UPDATETIME| 19M| 152M|18204 (3)|00:03:39| ------------------------------------------------------------------------------------------ 统计信息 ---------------------------------------------------------- 1 recursivecalls 0 dbblockgets 82428 consistentgets 82393 physicalreads 0 redosize 419 bytessentviaSQL*Nettoclient 385 bytesreceivedviaSQL*Netfromclient 2 SQL*Netroundtripsto/fromclient 0 sorts(memory) 0 sorts(disk) 1 rowsprocessed
完。
原创文章。为了维护文章的版本一致、最新、可追溯,转载请注明: 转载自idouba
本文链接地址: 从Count看Oracle执行计划的选择
从Count看Oracle执行计划的选择,布布扣,bubuko.com
原文:http://www.cnblogs.com/douba/p/analysis-oracle-execute-plan-from-count.html