分页语句优化注意点(一):关注业务数据

前些日子优化过一个分页函数,回头整理的时候发现了一个问题。让我做了一周多的实验才找到原因:

相关表的信息:SUBACC_FUND_INFO表90w+条数据  PARTY_INFO表11条数据 SYS_SEQ_ID是主键。TRAN_TIME的选择性也非常高。PARTY_ID是表PARTY_INFO的主键

实验一:

DROP INDEX IDX_SUBACC_FUND_INFO;
CREATE INDEX IDX_SUBACC_FUND_INFO ON SUBACC_FUND_INFO(TRAN_TIME DESC,SYS_SEQ_ID DESC) ONLINE;
SELECT *
  FROM (SELECT /*+ GATHER_PLAN_STATISTICS INDEX(C,IDX_SUBACC_FUND_INFO) USE_NL(C,PI) LEADING(C) */
         C.*, PI.CN_NAME
          FROM SUBACC_FUND_INFO C, PARTY_INFO PI
         WHERE PI.PARTY_ID = C.PARTY_ID
         ORDER BY C.TRAN_TIME DESC, C.SYS_SEQ_ID DESC)
 WHERE ROWNUM <= 20;
SELECT * FROM TABLE(DBMS_XPLAN.display_cursor(NULL,NULL,'ALLSTATS LAST'));
-----------------------------------------------------------------------------------------------------------------
| Id  | Operation                      | Name                 | Starts | E-Rows | A-Rows |   A-Time   | Buffers |
-----------------------------------------------------------------------------------------------------------------
|*  1 |  COUNT STOPKEY                 |                      |      1 |        |     20 |00:00:00.01 |      30 |
|   2 |   VIEW                         |                      |      1 |  20676 |     20 |00:00:00.01 |      30 |
|   3 |    NESTED LOOPS                |                      |      1 |  20676 |     20 |00:00:00.01 |      30 |
|   4 |     TABLE ACCESS BY INDEX ROWID| SUBACC_FUND_INFO     |      1 |    914K|     20 |00:00:00.01 |      25 |
|   5 |      INDEX FULL SCAN           | IDX_SUBACC_FUND_INFO |      1 |    914K|     20 |00:00:00.01 |       5 |
|*  6 |     INDEX RANGE SCAN           | IDX_PARTY_INFO       |     20 |      1 |     20 |00:00:00.01 |       5 |
-----------------------------------------------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
   1 - filter(ROWNUM<=20)
   6 - access("PI"."PARTY_ID"="C"."PARTY_ID")

实验二:修改order by顺序和组合索引的顺序

DROP INDEX IDX_SUBACC_FUND_INFO;
CREATE INDEX IDX_SUBACC_FUND_INFO ON SUBACC_FUND_INFO(SYS_SEQ_ID DESC,TRAN_TIME DESC) ONLINE;
SELECT *
  FROM (SELECT /*+ GATHER_PLAN_STATISTICS INDEX(C,IDX_SUBACC_FUND_INFO) USE_NL(C,PI) LEADING(C) */
         C.*, PI.CN_NAME
          FROM SUBACC_FUND_INFO C, PARTY_INFO PI
         WHERE PI.PARTY_ID = C.PARTY_ID
         ORDER BY C.SYS_SEQ_ID DESC, C.TRAN_TIME DESC)
 WHERE ROWNUM <= 20;
SELECT * FROM TABLE(DBMS_XPLAN.display_cursor(NULL,NULL,'ALLSTATS LAST'));
-----------------------------------------------------------------------------------------------------------------
| Id  | Operation                      | Name                 | Starts | E-Rows | A-Rows |   A-Time   | Buffers |
-----------------------------------------------------------------------------------------------------------------
|*  1 |  COUNT STOPKEY                 |                      |      1 |        |     20 |00:00:05.98 |     782K|
|   2 |   VIEW                         |                      |      1 |  20676 |     20 |00:00:05.98 |     782K|
|   3 |    NESTED LOOPS                |                      |      1 |  20676 |     20 |00:00:05.98 |     782K|
|   4 |     TABLE ACCESS BY INDEX ROWID| SUBACC_FUND_INFO     |      1 |    914K|    852K|00:00:03.41 |     782K|
|   5 |      INDEX FULL SCAN           | IDX_SUBACC_FUND_INFO |      1 |    914K|    852K|00:00:00.01 |    5846 |
|*  6 |     INDEX RANGE SCAN           | IDX_PARTY_INFO       |    852K|      1 |     20 |00:00:02.03 |       6 |
-----------------------------------------------------------------------------------------------------------------
Predicate Information (identified by operation id):
---------------------------------------------------
   1 - filter(ROWNUM<=20)
   6 - access("PI"."PARTY_ID"="C"."PARTY_ID")
在预估的PLAN里面两种执行计划一模一样,在真实的执行计划中返回的真实行数A-Rows却是千差万别
由上面的SQL语句及创建索引的语句可以看出来无非就是ORDER BY两个字段的顺序变了,但是和索引保持一致
这种问题,我以前从未遇到,为了解决问题,我做了如下实验来猜想我哪一个环节出了问题
1.看字段名称和类型是否有影响。(类型设一致。名称互换。结果:无影响)
2.看字段的长度是否有影响。(长度设置成一样、长度互换。结果:无影响)
3.看字段内容的长度是否有影响。(字段长度设置成一样、字段长度互换。结果:无影响)
4.去掉关联表,只对一个表分页。(结果:有影响)实验如下 :

DROP INDEX IDX_SUBACC_FUND_INFO;
CREATE INDEX IDX_SUBACC_FUND_INFO ON SUBACC_FUND_INFO(TRAN_TIME DESC,SYS_SEQ_ID DESC) ONLINE;
SELECT *
  FROM (SELECT *
          FROM SUBACC_FUND_INFO C
         ORDER BY C.TRAN_TIME DESC, C.SYS_SEQ_ID DESC)
 WHERE ROWNUM <= 20;
--结果:0.1S响应,逻辑读24
DROP INDEX IDX_SUBACC_FUND_INFO;
CREATE INDEX IDX_SUBACC_FUND_INFO ON SUBACC_FUND_INFO(SYS_SEQ_ID DESC,TRAN_TIME DESC) ONLINE;
SELECT *
  FROM (SELECT *
          FROM SUBACC_FUND_INFO C
         ORDER BY  C.SYS_SEQ_ID DESC,C.TRAN_TIME DESC)
 WHERE ROWNUM <= 20;
--结果:0.1S响应,逻辑读24

由此得出。可能是关联表PARTY_INFO的原因:

SELECT A.PARTY_ID,A.EN_NAME,A.CN_NAME,A.ADDRESS FROM PARTY_INFO A


可以看出上面的数据很乱。根本就是手工输入进去的测试数据,也就是说有很多数据很可能都匹配不到结果
所以我猜测实验2里面,排序之后的前N行(N<=1000)都匹配不到数据。
我们看真实的plan里面返回852K就,大胆的猜测一下:可能索引返回852K行的时候才在PI表中匹配到数据

对实验一验证:

SELECT CC.PARTY_ID, SYS_SEQ_ID, TRAN_TIME, PI.PARTY_ID,NUM1 "行数"
  FROM (SELECT PARTY_ID,
               SYS_SEQ_ID,
               TRAN_TIME,
               ROW_NUMBER() OVER(ORDER BY C.TRAN_TIME DESC, C.SYS_SEQ_ID DESC) NUM1
          FROM SUBACC_FUND_INFO C) CC,
       PARTY_INFO PI
 WHERE PI.PARTY_ID = CC.PARTY_ID;

对实验二的验证:

SELECT CC.PARTY_ID, SYS_SEQ_ID, TRAN_TIME, PI.PARTY_ID, NUM1 "行数"
  FROM (SELECT PARTY_ID,
               SYS_SEQ_ID,
               TRAN_TIME,
               ROW_NUMBER() OVER(ORDER BY C.SYS_SEQ_ID DESC, C.TRAN_TIME DESC) NUM1
          FROM SUBACC_FUND_INFO C) CC,
       PARTY_INFO PI
 WHERE PI.PARTY_ID = CC.PARTY_ID;

由上可得:实验1索引返回数据的前20条都在参数表PI中能够匹配到,扫描20条数据就停止。所以能够很快响应
          实验2索引返回的数据前20条都没有能在参数表PI中匹配到,所以只能往下继续扫描第21,22,23,……一直到852824行的时候才开始匹配到结果
          也就是说实验2的分页是从第852824行开始取20条数据返回。这就是坑爹的原因!!!!
 我们可能更多的是去尝试怎么优化,而很少去关注业务数据。所以有时候我们还是需要稍微的关注一下业务数据!
其实正常情况下是不会出现这种问题的,因为业务的明细数据去关联参数表(维度表),都是100%可以匹配上的。这是最坑爹的地方就是测试环境的数据有问题
但是有种类似情况的分页语句优化还是需要引起注意:
比如:分页函数里面的表关联的结果返回只有19条数据,而rownum <= 20。这种情况比上述情况还要恶劣,因为不够20行就会一直扫描,直到扫描驱动表所有的行
      这种结果在多表关联的分页函数里面还是会出现的。这种优化方法也很简单就是改SQL:rownum <= 20 改成 rownum <= 19 。搞定!!!


猜你喜欢

转载自blog.csdn.net/skybig1988/article/details/46670321