顺达主管2540437[20201210]sql语句优化.txt

来源:这里教程网 时间:2026-03-03 16:18:09 作者:

1.环境:   xxxxx> @ ver1   PORT_STRING                    VERSION        BANNER   ------------------------------ -------------- --------------------------------------------------------------------------------   x86_64/Linux 2.4.xx            11.2.0.4.0     Oracle Database 11g Enterprise Edition Release 11.2.0.4.0 - 64bit Production   xxxxx> @ sqlid bw49msgp0a9sn   SQL_ID        SQLTEXT   ------------- ------------------------------------------------------------------------------------------------------------------   bw49msgp0a9sn SELECT XSYS FROM ZY_CWTJ WHERE JLXH =( SELECT MAX ( JLXH ) FROM ZY_CWTJ WHERE BRKS =:1 AND BQPB =0 AND CZRQ < :2 )   xxxxx> @ dpc bw49msgp0a9sn ''   PLAN_TABLE_OUTPUT   -------------------------------------   SQL_ID  bw49msgp0a9sn, child number 0   -------------------------------------   SELECT XSYS FROM ZY_CWTJ WHERE JLXH =( SELECT MAX ( JLXH ) FROM ZY_CWTJ   WHERE BRKS =:1 AND BQPB =0 AND CZRQ < :2 )   Plan hash value: 1158680521   -------------------------------------------------------------------------------------------   | Id  | Operation                   | Name       | E-Rows |E-Bytes| Cost (%CPU)| E-Time   |   -------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT            |            |        |       |     3 (100)|          |   |   1 |  TABLE ACCESS BY INDEX ROWID| ZY_CWTJ    |      1 |    10 |     3   (0)| 00:00:01 |   |*  2 |   INDEX UNIQUE SCAN         | PK_ZY_CWTJ |      1 |       |     2   (0)| 00:00:01 |   |   3 |    SORT AGGREGATE           |            |      1 |    21 |            |          |   |*  4 |     TABLE ACCESS FULL       | ZY_CWTJ    |   7518 |   154K|  2409   (1)| 00:00:29 |   -------------------------------------------------------------------------------------------   Query Block Name / Object Alias (identified by operation id):   -------------------------------------------------------------      1 - SEL$1 / ZY_CWTJ@SEL$1      2 - SEL$1 / ZY_CWTJ@SEL$1      3 - SEL$2      4 - SEL$2 / ZY_CWTJ@SEL$2   Peeked Binds (identified by position):   --------------------------------------      1 - (NUMBER): 82      2 - (DATE): 12/09/2020 00:03:19   Predicate Information (identified by operation id):   ---------------------------------------------------      2 - access("JLXH"=)      4 - filter(("BRKS"=:1 AND "BQPB"=0 AND "CZRQ"<:2))   --//已经讲了N多次,开发为什么不仔细想想,这样要扫描ZY_CWTJ2次。另外说明一下复合索引BRKS,BQPB是存在的,实际上选择性很差,   --//基本不会使用这个的索引,我一上来就删除了。   --//另外我真心不愿意提交任何修改报告,我们这里业务流程不复杂,但是时间N久才完成。首先填写工单,而且提交后要等N久,而且我   --//还必须跟踪催促相关人员,我真心不知道我在修好修改语句的情况下,为什么需要很长时间完成修改任务。   --//如果不催促,这样的修改往往是遥遥无期。   xxxxx> @ bind_cap bw49msgp0a9sn ''   C200   ------------------------------------------------------------------------------------------------------------------   SELECT XSYS FROM ZY_CWTJ WHERE JLXH =( SELECT MAX ( JLXH ) FROM ZY_CWTJ WHERE BRKS =:1 AND BQPB =0 AND CZRQ < :2 )   SQL_ID        CHILD_NUMBER WAS NAME POSITION MAX_LENGTH LAST_CAPTURED       DATATYPE_STRING VALUE_STRING        C30   ------------- ------------ --- ---- -------- ---------- ------------------- --------------- ------------------- ------------------------------   bw49msgp0a9sn            0 YES :1          1         22 2020-12-10 00:05:00 NUMBER          82                              YES :2          2          7 2020-12-10 00:05:00 DATE            2020/12/10 00:03:18   --//注意查询的绑定变量,实际上这个程序是一个job调用的,在凌晨统计信息的一部分。   2.首先我先写出我改写语句:   SELECT XSYS     FROM (  SELECT XSYS, JLXH               FROM ZY_CWTJ              WHERE BRKS = :1 AND BQPB = 0 AND CZRQ < :2           ORDER BY JLXH DESC)    WHERE ROWNUM = 1;   --//这样仅仅扫描1次。看看执行计划,为了了解sql执行情况加入提示gather_plan_statistics。另外说明一下JLXH是主键,顺序递增,   --//上面的语句仅仅会返回1条,不会出现歧义性。执行计划如下:   SQL_ID  4kkk01y9tzmuc, child number 0   -------------------------------------   SELECT /*+ gather_plan_statistics */  XSYS   FROM (  SELECT XSYS, JLXH                FROM ZY_CWTJ            WHERE BRKS = :1 AND BQPB = 0 AND   CZRQ < :2         ORDER BY JLXH DESC)  WHERE ROWNUM = 1   Plan hash value: 1497600757   --------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                     | Name       | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   --------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT              |            |      1 |        |       |    26 (100)|          |      1 |00:00:00.01 |       6 |   |*  1 |  COUNT STOPKEY                |            |      1 |        |       |            |          |      1 |00:00:00.01 |       6 |   |   2 |   VIEW                        |            |      1 |      2 |    26 |    26   (0)| 00:00:01 |      1 |00:00:00.01 |       6 |   |*  3 |    TABLE ACCESS BY INDEX ROWID| ZY_CWTJ    |      1 |     20 |   480 |    26   (0)| 00:00:01 |      1 |00:00:00.01 |       6 |   |   4 |     INDEX FULL SCAN DESCENDING| PK_ZY_CWTJ |      1 |   3120 |       |     8   (0)| 00:00:01 |    400 |00:00:00.01 |       3 |   --------------------------------------------------------------------------------------------------------------------------------------   Query Block Name / Object Alias (identified by operation id):   -------------------------------------------------------------      1 - SEL$1      2 - SEL$2 / from$_subquery$_001@SEL$1      3 - SEL$2 / ZY_CWTJ@SEL$2      4 - SEL$2 / ZY_CWTJ@SEL$2   Outline Data   -------------     /*+         BEGIN_OUTLINE_DATA         IGNORE_OPTIM_EMBEDDED_HINTS         OPTIMIZER_FEATURES_ENABLE('11.2.0.4')         DB_VERSION('11.2.0.4')         OPT_PARAM('_optim_peek_user_binds' 'false')   ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~               OPT_PARAM('_bloom_filter_enabled' 'false')   ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~==>不知道安装者要修改这2个隐含参数。               ALL_ROWS         OUTLINE_LEAF(@"SEL$2")         OUTLINE_LEAF(@"SEL$1")         NO_ACCESS(@"SEL$1" "from$_subquery$_001"@"SEL$1")         INDEX_DESC(@"SEL$2" "ZY_CWTJ"@"SEL$2" ("ZY_CWTJ"."JLXH"))         END_OUTLINE_DATA     */   Predicate Information (identified by operation id):   ---------------------------------------------------      1 - filter(ROWNUM=1)      3 - filter(("BRKS"=:1 AND "CZRQ"<:2 AND "BQPB"=0))   Column Projection Information (identified by operation id):   -----------------------------------------------------------      1 - "XSYS"[NUMBER,22]      2 - "XSYS"[NUMBER,22]      3 - "XSYS"[NUMBER,22]      4 - "ZY_CWTJ".ROWID[ROWID,10]   --//你可以发现逻辑读仅仅6个,而且我并没有建立新的索引,直接利用主键JLXH的索引。   --//因为查询条件非常特殊,总是查询的日期CZRQ是执行语句的当前时间,也就是表中的最大值。查看绑定变量就清楚了。jLXH是主键,顺序增加。   --//缺点就是如果查询根本不存在的BRKS值,这样就逻辑读很大,要扫描整个索引。我发现这样的情况还真心存在。   --//我估计分院与总院的科室表都是一样的,这样两边的统计也需要扫描整个科室列表。   --//比如BRKS 带入 10000,执行计划的逻辑读上升,执行计划如下:   Plan hash value: 1497600757   --------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                     | Name       | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   --------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT              |            |      1 |        |       |    26 (100)|          |      0 |00:00:00.01 |     241 |   |*  1 |  COUNT STOPKEY                |            |      1 |        |       |            |          |      0 |00:00:00.01 |     241 |   |   2 |   VIEW                        |            |      1 |      2 |    26 |    26   (0)| 00:00:01 |      0 |00:00:00.01 |     241 |   |*  3 |    TABLE ACCESS BY INDEX ROWID| ZY_CWTJ    |      1 |     20 |   480 |    26   (0)| 00:00:01 |      0 |00:00:00.01 |     241 |   |   4 |     INDEX FULL SCAN DESCENDING| PK_ZY_CWTJ |      1 |   3120 |       |     8   (0)| 00:00:01 |  31262 |00:00:00.01 |      64 |   --------------------------------------------------------------------------------------------------------------------------------------   --//基本扫描整个索引,再探查整个表,效率比全表扫描要差许多。看全表扫描的执行计划:   --//这样情况出现逻辑读依旧很大,可以建立这样的索引,充分利用取jlxh最大的特性。   create unique index I_ZY_CWTJ_BRKS_BQPB_JLXH on zy_cwtj (brks, bqpb, jlxh) logging compress 2;   Plan hash value: 2516239191   -----------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name                     | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -----------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                          |      1 |        |       |     4 (100)|          |      1 |00:00:00.01 |       5 |   |*  1 |  COUNT STOPKEY                 |                          |      1 |        |       |            |          |      1 |00:00:00.01 |       5 |   |   2 |   VIEW                         |                          |      1 |      2 |    26 |     4   (0)| 00:00:01 |      1 |00:00:00.01 |       5 |   |*  3 |    TABLE ACCESS BY INDEX ROWID | ZY_CWTJ                  |      1 |     20 |   480 |     4   (0)| 00:00:01 |      1 |00:00:00.01 |       5 |   |*  4 |     INDEX RANGE SCAN DESCENDING| I_ZY_CWTJ_BRKS_BQPB_JLXH |      1 |     40 |       |     2   (0)| 00:00:01 |     21 |00:00:00.01 |       2 |   -----------------------------------------------------------------------------------------------------------------------------------------------------   --//这样即可以保持索引很小,查询逻辑读很低。如果查询CZRQ 条件是 '2019/12/8 16:11:15',这样逻辑读会很高。   Plan hash value: 2516239191   -----------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name                     | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -----------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                          |      1 |        |       |     4 (100)|          |      0 |00:00:00.01 |     171 |   |*  1 |  COUNT STOPKEY                 |                          |      1 |        |       |            |          |      0 |00:00:00.01 |     171 |   |   2 |   VIEW                         |                          |      1 |      2 |    26 |     4   (0)| 00:00:01 |      0 |00:00:00.01 |     171 |   |*  3 |    TABLE ACCESS BY INDEX ROWID | ZY_CWTJ                  |      1 |     20 |   480 |     4   (0)| 00:00:01 |      0 |00:00:00.01 |     171 |   |*  4 |     INDEX RANGE SCAN DESCENDING| I_ZY_CWTJ_BRKS_BQPB_JLXH |      1 |     40 |       |     2   (0)| 00:00:01 |   1196 |00:00:00.01 |       5 |   -----------------------------------------------------------------------------------------------------------------------------------------------------   --//但是这样的查询在实际系统中基本不存在的。   3.回到生产系统看看原来的语句的这样的索引效果如何:   create unique index I_ZY_CWTJ_BRKS_BQPB_JLXH on zy_cwtj (brks, bqpb, jlxh) logging compress 2;   --//BRKS 带入 10000,执行计划如下:   -----------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name                     | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -----------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                          |      1 |        |       |     2 (100)|          |      0 |00:00:00.01 |       2 |   |   1 |  TABLE ACCESS BY INDEX ROWID   | ZY_CWTJ                  |      1 |      1 |     9 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |*  2 |   INDEX UNIQUE SCAN            | PK_ZY_CWTJ               |      1 |      1 |       |     1   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |   3 |    SORT AGGREGATE              |                          |      1 |      1 |    21 |            |          |      1 |00:00:00.01 |       2 |   |*  4 |     TABLE ACCESS BY INDEX ROWID| ZY_CWTJ                  |      1 |     20 |   420 |    13   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |*  5 |      INDEX RANGE SCAN          | I_ZY_CWTJ_BRKS_BQPB_JLXH |      1 |    401 |       |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   -----------------------------------------------------------------------------------------------------------------------------------------------------   --//如果带入存在的BRKS值,逻辑读反而上升。   Plan hash value: 353781330   -----------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name                     | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -----------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                          |      1 |        |       |     2 (100)|          |      1 |00:00:00.01 |     174 |   |   1 |  TABLE ACCESS BY INDEX ROWID   | ZY_CWTJ                  |      1 |      1 |     9 |     2   (0)| 00:00:01 |      1 |00:00:00.01 |     174 |   |*  2 |   INDEX UNIQUE SCAN            | PK_ZY_CWTJ               |      1 |      1 |       |     1   (0)| 00:00:01 |      1 |00:00:00.01 |     173 |   |   3 |    SORT AGGREGATE              |                          |      1 |      1 |    21 |            |          |      1 |00:00:00.01 |     171 |   |*  4 |     TABLE ACCESS BY INDEX ROWID| ZY_CWTJ                  |      1 |     20 |   420 |    13   (0)| 00:00:01 |   1199 |00:00:00.01 |     171 |   |*  5 |      INDEX RANGE SCAN          | I_ZY_CWTJ_BRKS_BQPB_JLXH |      1 |    401 |       |     2   (0)| 00:00:01 |   1199 |00:00:00.01 |       5 |   -----------------------------------------------------------------------------------------------------------------------------------------------------   --//禁用上面的索引,设置该索引隐含。继续测试:   create index i_zy_cwtj_laji on zy_cwtj (brks, bqpb, czrq, jlxh) compress 2;   SELECT /*+ gather_plan_statistics  */  XSYS   FROM (  SELECT  XSYS,   JLXH             FROM ZY_CWTJ            WHERE BRKS = :1 AND BQPB = 0   AND CZRQ < :2         ORDER BY JLXH DESC)  WHERE ROWNUM = 1       Plan hash value: 4136658378   ----------------------------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name           | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |  OMem |  1Mem | Used-Mem |   ----------------------------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                |      1 |        |       |     4 (100)|          |      1 |00:00:00.01 |     171 |       |       |          |   |*  1 |  COUNT STOPKEY                 |                |      1 |        |       |            |          |      1 |00:00:00.01 |     171 |       |       |          |   |   2 |   VIEW                         |                |      1 |     20 |   260 |     4  (25)| 00:00:01 |      1 |00:00:00.01 |     171 |       |       |          |   |*  3 |    SORT ORDER BY STOPKEY       |                |      1 |     20 |   480 |     4  (25)| 00:00:01 |      1 |00:00:00.01 |     171 |  2048 |  2048 | 2048  (0)|   |   4 |     TABLE ACCESS BY INDEX ROWID| ZY_CWTJ        |      1 |     20 |   480 |     3   (0)| 00:00:01 |   1176 |00:00:00.01 |     171 |       |       |          |   |*  5 |      INDEX RANGE SCAN          | I_ZY_CWTJ_LAJI |      1 |      4 |       |     2   (0)| 00:00:01 |   1176 |00:00:00.01 |       7 |       |       |          |   ----------------------------------------------------------------------------------------------------------------------------------------------------------------------   --//可以发现这样的索引并不佳,逻辑读很高。而利用它取jlxh最大值,以及查询CZRQ值都是最大值的特点。重新建立索引如下:   create index i_zy_cwtj_laji on zy_cwtj (brks, bqpb, jlxh,czrq ) compress 2;   --//索引czrq,jlxh交换对比前面的索引。   --//查询存在的BRKS情况。   SQL_ID  8rvdqyj2pj5kp, child number 0   -------------------------------------   SELECT /*+ gather_plan_statistics */  XSYS   FROM ZY_CWTJ  WHERE JLXH =   (SELECT MAX (JLXH)                  FROM ZY_CWTJ                 WHERE   BRKS = :1 AND BQPB = 0 AND CZRQ < :2)   Plan hash value: 1977445073   ----------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name           | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers | Reads  |   ----------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                |      1 |        |       |     2 (100)|          |      1 |00:00:00.01 |       5 |      1 |   |   1 |  TABLE ACCESS BY INDEX ROWID   | ZY_CWTJ        |      1 |      1 |     9 |     2   (0)| 00:00:01 |      1 |00:00:00.01 |       5 |      1 |   |*  2 |   INDEX UNIQUE SCAN            | PK_ZY_CWTJ     |      1 |      1 |       |     1   (0)| 00:00:01 |      1 |00:00:00.01 |       4 |      1 |   |   3 |    SORT AGGREGATE              |                |      1 |      1 |    21 |            |          |      1 |00:00:00.01 |       2 |      1 |   |   4 |     FIRST ROW                  |                |      1 |      1 |    21 |     2   (0)| 00:00:01 |      1 |00:00:00.01 |       2 |      1 |   |*  5 |      INDEX RANGE SCAN (MIN/MAX)| I_ZY_CWTJ_LAJI |      1 |      1 |    21 |     2   (0)| 00:00:01 |      1 |00:00:00.01 |       2 |      1 |   ----------------------------------------------------------------------------------------------------------------------------------------------------   --//这样效果最好,因为查询条件 CZRQ < :2,不是等值查询。注意执行计划出现INDEX RANGE SCAN (MIN/MAX).   --//如果查询BRKS在表中不存在,带入10000.   -------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name           | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                |      1 |        |       |     2 (100)|          |      0 |00:00:00.01 |       2 |   |   1 |  TABLE ACCESS BY INDEX ROWID   | ZY_CWTJ        |      1 |      1 |     9 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |*  2 |   INDEX UNIQUE SCAN            | PK_ZY_CWTJ     |      1 |      1 |       |     1   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |   3 |    SORT AGGREGATE              |                |      1 |      1 |    21 |            |          |      1 |00:00:00.01 |       2 |   |   4 |     FIRST ROW                  |                |      1 |      1 |    21 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |*  5 |      INDEX RANGE SCAN (MIN/MAX)| I_ZY_CWTJ_LAJI |      1 |      1 |    21 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   -------------------------------------------------------------------------------------------------------------------------------------------   --//两者差异不大。   --//这样效果最好,因为查询条件 CZRQ < :2,不是等值查询。注意执行计划出现INDEX RANGE SCAN (MIN/MAX).而不是前面的INDEX RANGE SCAN。   --//即使我日期带入2018/12/10 0:03:18,逻辑读也不会上升很大。   -------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name           | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                |      1 |        |       |     2 (100)|          |      0 |00:00:00.01 |       6 |   |   1 |  TABLE ACCESS BY INDEX ROWID   | ZY_CWTJ        |      1 |      1 |     9 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       6 |   |*  2 |   INDEX UNIQUE SCAN            | PK_ZY_CWTJ     |      1 |      1 |       |     1   (0)| 00:00:01 |      0 |00:00:00.01 |       6 |   |   3 |    SORT AGGREGATE              |                |      1 |      1 |    21 |            |          |      1 |00:00:00.01 |       6 |   |   4 |     FIRST ROW                  |                |      1 |      1 |    21 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       6 |   |*  5 |      INDEX RANGE SCAN (MIN/MAX)| I_ZY_CWTJ_LAJI |      1 |      1 |    21 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       6 |   -------------------------------------------------------------------------------------------------------------------------------------------   Predicate Information (identified by operation id):   ---------------------------------------------------      2 - access("JLXH"=)      5 - access("BRKS"=:1 AND "BQPB"=0)          filter("CZRQ"<:2)   --//注意Predicate Information部分,仅仅在索引中探查。  1.环境:    

xxxxx> @ ver1   PORT_STRING                    VERSION        BANNER   ------------------------------ -------------- --------------------------------------------------------------------------------   x86_64/Linux 2.4.xx            11.2.0.4.0     Oracle Database 11g Enterprise Edition Release 11.2.0.4.0 - 64bit Production   xxxxx> @ sqlid bw49msgp0a9sn   SQL_ID        SQLTEXT   ------------- ------------------------------------------------------------------------------------------------------------------   bw49msgp0a9sn SELECT XSYS FROM ZY_CWTJ WHERE JLXH =( SELECT MAX ( JLXH ) FROM ZY_CWTJ WHERE BRKS =:1 AND BQPB =0 AND CZRQ < :2 )   xxxxx> @ dpc bw49msgp0a9sn ''   PLAN_TABLE_OUTPUT   -------------------------------------   SQL_ID  bw49msgp0a9sn, child number 0   -------------------------------------   SELECT XSYS FROM ZY_CWTJ WHERE JLXH =( SELECT MAX ( JLXH ) FROM ZY_CWTJ   WHERE BRKS =:1 AND BQPB =0 AND CZRQ < :2 )   Plan hash value: 1158680521   -------------------------------------------------------------------------------------------   | Id  | Operation                   | Name       | E-Rows |E-Bytes| Cost (%CPU)| E-Time   |   -------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT            |            |        |       |     3 (100)|          |   |   1 |  TABLE ACCESS BY INDEX ROWID| ZY_CWTJ    |      1 |    10 |     3   (0)| 00:00:01 |   |*  2 |   INDEX UNIQUE SCAN         | PK_ZY_CWTJ |      1 |       |     2   (0)| 00:00:01 |   |   3 |    SORT AGGREGATE           |            |      1 |    21 |            |          |   |*  4 |     TABLE ACCESS FULL       | ZY_CWTJ    |   7518 |   154K|  2409   (1)| 00:00:29 |   -------------------------------------------------------------------------------------------   Query Block Name / Object Alias (identified by operation id):   -------------------------------------------------------------      1 - SEL$1 / ZY_CWTJ@SEL$1      2 - SEL$1 / ZY_CWTJ@SEL$1      3 - SEL$2      4 - SEL$2 / ZY_CWTJ@SEL$2   Peeked Binds (identified by position):   --------------------------------------      1 - (NUMBER): 82      2 - (DATE): 12/09/2020 00:03:19   Predicate Information (identified by operation id):   ---------------------------------------------------      2 - access("JLXH"=)      4 - filter(("BRKS"=:1 AND "BQPB"=0 AND "CZRQ"<:2))   --//已经讲了N多次,开发为什么不仔细想想,这样要扫描ZY_CWTJ2次。另外说明一下复合索引BRKS,BQPB是存在的,实际上选择性很差,   --//基本不会使用这个的索引,我一上来就删除了。   --//另外我真心不愿意提交任何修改报告,我们这里业务流程不复杂,但是时间N久才完成。首先填写工单,而且提交后要等N久,而且我   --//还必须跟踪催促相关人员,我真心不知道我在修好修改语句的情况下,为什么需要很长时间完成修改任务。   --//如果不催促,这样的修改往往是遥遥无期。   xxxxx> @ bind_cap bw49msgp0a9sn ''   C200   ------------------------------------------------------------------------------------------------------------------   SELECT XSYS FROM ZY_CWTJ WHERE JLXH =( SELECT MAX ( JLXH ) FROM ZY_CWTJ WHERE BRKS =:1 AND BQPB =0 AND CZRQ < :2 )   SQL_ID        CHILD_NUMBER WAS NAME POSITION MAX_LENGTH LAST_CAPTURED       DATATYPE_STRING VALUE_STRING        C30   ------------- ------------ --- ---- -------- ---------- ------------------- --------------- ------------------- ------------------------------   bw49msgp0a9sn            0 YES :1          1         22 2020-12-10 00:05:00 NUMBER          82                              YES :2          2          7 2020-12-10 00:05:00 DATE            2020/12/10 00:03:18   --//注意查询的绑定变量,实际上这个程序是一个job调用的,在凌晨统计信息的一部分。   2.首先我先写出我改写语句:   SELECT XSYS     FROM (  SELECT XSYS, JLXH               FROM ZY_CWTJ              WHERE BRKS = :1 AND BQPB = 0 AND CZRQ < :2           ORDER BY JLXH DESC)    WHERE ROWNUM = 1;   --//这样仅仅扫描1次。看看执行计划,为了了解sql执行情况加入提示gather_plan_statistics。另外说明一下JLXH是主键,顺序递增,   --//上面的语句仅仅会返回1条,不会出现歧义性。执行计划如下:   SQL_ID  4kkk01y9tzmuc, child number 0   -------------------------------------   SELECT /*+ gather_plan_statistics */  XSYS   FROM (  SELECT XSYS, JLXH                FROM ZY_CWTJ            WHERE BRKS = :1 AND BQPB = 0 AND   CZRQ < :2         ORDER BY JLXH DESC)  WHERE ROWNUM = 1   Plan hash value: 1497600757   --------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                     | Name       | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   --------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT              |            |      1 |        |       |    26 (100)|          |      1 |00:00:00.01 |       6 |   |*  1 |  COUNT STOPKEY                |            |      1 |        |       |            |          |      1 |00:00:00.01 |       6 |   |   2 |   VIEW                        |            |      1 |      2 |    26 |    26   (0)| 00:00:01 |      1 |00:00:00.01 |       6 |   |*  3 |    TABLE ACCESS BY INDEX ROWID| ZY_CWTJ    |      1 |     20 |   480 |    26   (0)| 00:00:01 |      1 |00:00:00.01 |       6 |   |   4 |     INDEX FULL SCAN DESCENDING| PK_ZY_CWTJ |      1 |   3120 |       |     8   (0)| 00:00:01 |    400 |00:00:00.01 |       3 |   --------------------------------------------------------------------------------------------------------------------------------------   Query Block Name / Object Alias (identified by operation id):   -------------------------------------------------------------      1 - SEL$1      2 - SEL$2 / from$_subquery$_001@SEL$1      3 - SEL$2 / ZY_CWTJ@SEL$2      4 - SEL$2 / ZY_CWTJ@SEL$2   Outline Data   -------------     /*+         BEGIN_OUTLINE_DATA         IGNORE_OPTIM_EMBEDDED_HINTS         OPTIMIZER_FEATURES_ENABLE('11.2.0.4')         DB_VERSION('11.2.0.4')         OPT_PARAM('_optim_peek_user_binds' 'false')   ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~               OPT_PARAM('_bloom_filter_enabled' 'false')   ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~==>不知道安装者要修改这2个隐含参数。               ALL_ROWS         OUTLINE_LEAF(@"SEL$2")         OUTLINE_LEAF(@"SEL$1")         NO_ACCESS(@"SEL$1" "from$_subquery$_001"@"SEL$1")         INDEX_DESC(@"SEL$2" "ZY_CWTJ"@"SEL$2" ("ZY_CWTJ"."JLXH"))         END_OUTLINE_DATA     */   Predicate Information (identified by operation id):   ---------------------------------------------------      1 - filter(ROWNUM=1)      3 - filter(("BRKS"=:1 AND "CZRQ"<:2 AND "BQPB"=0))   Column Projection Information (identified by operation id):   -----------------------------------------------------------      1 - "XSYS"[NUMBER,22]      2 - "XSYS"[NUMBER,22]      3 - "XSYS"[NUMBER,22]      4 - "ZY_CWTJ".ROWID[ROWID,10]   --//你可以发现逻辑读仅仅6个,而且我并没有建立新的索引,直接利用主键JLXH的索引。   --//因为查询条件非常特殊,总是查询的日期CZRQ是执行语句的当前时间,也就是表中的最大值。查看绑定变量就清楚了。jLXH是主键,顺序增加。   --//缺点就是如果查询根本不存在的BRKS值,这样就逻辑读很大,要扫描整个索引。我发现这样的情况还真心存在。   --//我估计分院与总院的科室表都是一样的,这样两边的统计也需要扫描整个科室列表。   --//比如BRKS 带入 10000,执行计划的逻辑读上升,执行计划如下:   Plan hash value: 1497600757   --------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                     | Name       | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   --------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT              |            |      1 |        |       |    26 (100)|          |      0 |00:00:00.01 |     241 |   |*  1 |  COUNT STOPKEY                |            |      1 |        |       |            |          |      0 |00:00:00.01 |     241 |   |   2 |   VIEW                        |            |      1 |      2 |    26 |    26   (0)| 00:00:01 |      0 |00:00:00.01 |     241 |   |*  3 |    TABLE ACCESS BY INDEX ROWID| ZY_CWTJ    |      1 |     20 |   480 |    26   (0)| 00:00:01 |      0 |00:00:00.01 |     241 |   |   4 |     INDEX FULL SCAN DESCENDING| PK_ZY_CWTJ |      1 |   3120 |       |     8   (0)| 00:00:01 |  31262 |00:00:00.01 |      64 |   --------------------------------------------------------------------------------------------------------------------------------------   --//基本扫描整个索引,再探查整个表,效率比全表扫描要差许多。看全表扫描的执行计划:   --//这样情况出现逻辑读依旧很大,可以建立这样的索引,充分利用取jlxh最大的特性。   create unique index I_ZY_CWTJ_BRKS_BQPB_JLXH on zy_cwtj (brks, bqpb, jlxh) logging compress 2;   Plan hash value: 2516239191   -----------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name                     | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -----------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                          |      1 |        |       |     4 (100)|          |      1 |00:00:00.01 |       5 |   |*  1 |  COUNT STOPKEY                 |                          |      1 |        |       |            |          |      1 |00:00:00.01 |       5 |   |   2 |   VIEW                         |                          |      1 |      2 |    26 |     4   (0)| 00:00:01 |      1 |00:00:00.01 |       5 |   |*  3 |    TABLE ACCESS BY INDEX ROWID | ZY_CWTJ                  |      1 |     20 |   480 |     4   (0)| 00:00:01 |      1 |00:00:00.01 |       5 |   |*  4 |     INDEX RANGE SCAN DESCENDING| I_ZY_CWTJ_BRKS_BQPB_JLXH |      1 |     40 |       |     2   (0)| 00:00:01 |     21 |00:00:00.01 |       2 |   -----------------------------------------------------------------------------------------------------------------------------------------------------   --//这样即可以保持索引很小,查询逻辑读很低。如果查询CZRQ 条件是 '2019/12/8 16:11:15',这样逻辑读会很高。   Plan hash value: 2516239191   -----------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name                     | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -----------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                          |      1 |        |       |     4 (100)|          |      0 |00:00:00.01 |     171 |   |*  1 |  COUNT STOPKEY                 |                          |      1 |        |       |            |          |      0 |00:00:00.01 |     171 |   |   2 |   VIEW                         |                          |      1 |      2 |    26 |     4   (0)| 00:00:01 |      0 |00:00:00.01 |     171 |   |*  3 |    TABLE ACCESS BY INDEX ROWID | ZY_CWTJ                  |      1 |     20 |   480 |     4   (0)| 00:00:01 |      0 |00:00:00.01 |     171 |   |*  4 |     INDEX RANGE SCAN DESCENDING| I_ZY_CWTJ_BRKS_BQPB_JLXH |      1 |     40 |       |     2   (0)| 00:00:01 |   1196 |00:00:00.01 |       5 |   -----------------------------------------------------------------------------------------------------------------------------------------------------   --//但是这样的查询在实际系统中基本不存在的。   3.回到生产系统看看原来的语句的这样的索引效果如何:   create unique index I_ZY_CWTJ_BRKS_BQPB_JLXH on zy_cwtj (brks, bqpb, jlxh) logging compress 2;   --//BRKS 带入 10000,执行计划如下:   -----------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name                     | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -----------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                          |      1 |        |       |     2 (100)|          |      0 |00:00:00.01 |       2 |   |   1 |  TABLE ACCESS BY INDEX ROWID   | ZY_CWTJ                  |      1 |      1 |     9 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |*  2 |   INDEX UNIQUE SCAN            | PK_ZY_CWTJ               |      1 |      1 |       |     1   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |   3 |    SORT AGGREGATE              |                          |      1 |      1 |    21 |            |          |      1 |00:00:00.01 |       2 |   |*  4 |     TABLE ACCESS BY INDEX ROWID| ZY_CWTJ                  |      1 |     20 |   420 |    13   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |*  5 |      INDEX RANGE SCAN          | I_ZY_CWTJ_BRKS_BQPB_JLXH |      1 |    401 |       |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   -----------------------------------------------------------------------------------------------------------------------------------------------------   --//如果带入存在的BRKS值,逻辑读反而上升。   Plan hash value: 353781330   -----------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name                     | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -----------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                          |      1 |        |       |     2 (100)|          |      1 |00:00:00.01 |     174 |   |   1 |  TABLE ACCESS BY INDEX ROWID   | ZY_CWTJ                  |      1 |      1 |     9 |     2   (0)| 00:00:01 |      1 |00:00:00.01 |     174 |   |*  2 |   INDEX UNIQUE SCAN            | PK_ZY_CWTJ               |      1 |      1 |       |     1   (0)| 00:00:01 |      1 |00:00:00.01 |     173 |   |   3 |    SORT AGGREGATE              |                          |      1 |      1 |    21 |            |          |      1 |00:00:00.01 |     171 |   |*  4 |     TABLE ACCESS BY INDEX ROWID| ZY_CWTJ                  |      1 |     20 |   420 |    13   (0)| 00:00:01 |   1199 |00:00:00.01 |     171 |   |*  5 |      INDEX RANGE SCAN          | I_ZY_CWTJ_BRKS_BQPB_JLXH |      1 |    401 |       |     2   (0)| 00:00:01 |   1199 |00:00:00.01 |       5 |   -----------------------------------------------------------------------------------------------------------------------------------------------------   --//禁用上面的索引,设置该索引隐含。继续测试:   create index i_zy_cwtj_laji on zy_cwtj (brks, bqpb, czrq, jlxh) compress 2;   SELECT /*+ gather_plan_statistics  */  XSYS   FROM (  SELECT  XSYS,   JLXH             FROM ZY_CWTJ            WHERE BRKS = :1 AND BQPB = 0   AND CZRQ < :2         ORDER BY JLXH DESC)  WHERE ROWNUM = 1       Plan hash value: 4136658378   ----------------------------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name           | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |  OMem |  1Mem | Used-Mem |   ----------------------------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                |      1 |        |       |     4 (100)|          |      1 |00:00:00.01 |     171 |       |       |          |   |*  1 |  COUNT STOPKEY                 |                |      1 |        |       |            |          |      1 |00:00:00.01 |     171 |       |       |          |   |   2 |   VIEW                         |                |      1 |     20 |   260 |     4  (25)| 00:00:01 |      1 |00:00:00.01 |     171 |       |       |          |   |*  3 |    SORT ORDER BY STOPKEY       |                |      1 |     20 |   480 |     4  (25)| 00:00:01 |      1 |00:00:00.01 |     171 |  2048 |  2048 | 2048  (0)|   |   4 |     TABLE ACCESS BY INDEX ROWID| ZY_CWTJ        |      1 |     20 |   480 |     3   (0)| 00:00:01 |   1176 |00:00:00.01 |     171 |       |       |          |   |*  5 |      INDEX RANGE SCAN          | I_ZY_CWTJ_LAJI |      1 |      4 |       |     2   (0)| 00:00:01 |   1176 |00:00:00.01 |       7 |       |       |          |   ----------------------------------------------------------------------------------------------------------------------------------------------------------------------   --//可以发现这样的索引并不佳,逻辑读很高。而利用它取jlxh最大值,以及查询CZRQ值都是最大值的特点。重新建立索引如下:   create index i_zy_cwtj_laji on zy_cwtj (brks, bqpb, jlxh,czrq ) compress 2;   --//索引czrq,jlxh交换对比前面的索引。   --//查询存在的BRKS情况。   SQL_ID  8rvdqyj2pj5kp, child number 0   -------------------------------------   SELECT /*+ gather_plan_statistics */  XSYS   FROM ZY_CWTJ  WHERE JLXH =   (SELECT MAX (JLXH)                  FROM ZY_CWTJ                 WHERE   BRKS = :1 AND BQPB = 0 AND CZRQ < :2)   Plan hash value: 1977445073   ----------------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name           | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers | Reads  |   ----------------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                |      1 |        |       |     2 (100)|          |      1 |00:00:00.01 |       5 |      1 |   |   1 |  TABLE ACCESS BY INDEX ROWID   | ZY_CWTJ        |      1 |      1 |     9 |     2   (0)| 00:00:01 |      1 |00:00:00.01 |       5 |      1 |   |*  2 |   INDEX UNIQUE SCAN            | PK_ZY_CWTJ     |      1 |      1 |       |     1   (0)| 00:00:01 |      1 |00:00:00.01 |       4 |      1 |   |   3 |    SORT AGGREGATE              |                |      1 |      1 |    21 |            |          |      1 |00:00:00.01 |       2 |      1 |   |   4 |     FIRST ROW                  |                |      1 |      1 |    21 |     2   (0)| 00:00:01 |      1 |00:00:00.01 |       2 |      1 |   |*  5 |      INDEX RANGE SCAN (MIN/MAX)| I_ZY_CWTJ_LAJI |      1 |      1 |    21 |     2   (0)| 00:00:01 |      1 |00:00:00.01 |       2 |      1 |   ----------------------------------------------------------------------------------------------------------------------------------------------------   --//这样效果最好,因为查询条件 CZRQ < :2,不是等值查询。注意执行计划出现INDEX RANGE SCAN (MIN/MAX).   --//如果查询BRKS在表中不存在,带入10000.   -------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name           | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                |      1 |        |       |     2 (100)|          |      0 |00:00:00.01 |       2 |   |   1 |  TABLE ACCESS BY INDEX ROWID   | ZY_CWTJ        |      1 |      1 |     9 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |*  2 |   INDEX UNIQUE SCAN            | PK_ZY_CWTJ     |      1 |      1 |       |     1   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |   3 |    SORT AGGREGATE              |                |      1 |      1 |    21 |            |          |      1 |00:00:00.01 |       2 |   |   4 |     FIRST ROW                  |                |      1 |      1 |    21 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   |*  5 |      INDEX RANGE SCAN (MIN/MAX)| I_ZY_CWTJ_LAJI |      1 |      1 |    21 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       2 |   -------------------------------------------------------------------------------------------------------------------------------------------   --//两者差异不大。   --//这样效果最好,因为查询条件 CZRQ < :2,不是等值查询。注意执行计划出现INDEX RANGE SCAN (MIN/MAX).而不是前面的INDEX RANGE SCAN。   --//即使我日期带入2018/12/10 0:03:18,逻辑读也不会上升很大。   -------------------------------------------------------------------------------------------------------------------------------------------   | Id  | Operation                      | Name           | Starts | E-Rows |E-Bytes| Cost (%CPU)| E-Time   | A-Rows |   A-Time   | Buffers |   -------------------------------------------------------------------------------------------------------------------------------------------   |   0 | SELECT STATEMENT               |                |      1 |        |       |     2 (100)|          |      0 |00:00:00.01 |       6 |   |   1 |  TABLE ACCESS BY INDEX ROWID   | ZY_CWTJ        |      1 |      1 |     9 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       6 |   |*  2 |   INDEX UNIQUE SCAN            | PK_ZY_CWTJ     |      1 |      1 |       |     1   (0)| 00:00:01 |      0 |00:00:00.01 |       6 |   |   3 |    SORT AGGREGATE              |                |      1 |      1 |    21 |            |          |      1 |00:00:00.01 |       6 |   |   4 |     FIRST ROW                  |                |      1 |      1 |    21 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       6 |   |*  5 |      INDEX RANGE SCAN (MIN/MAX)| I_ZY_CWTJ_LAJI |      1 |      1 |    21 |     2   (0)| 00:00:01 |      0 |00:00:00.01 |       6 |   -------------------------------------------------------------------------------------------------------------------------------------------   Predicate Information (identified by operation id):   ---------------------------------------------------      2 - access("JLXH"=)      5 - access("BRKS"=:1 AND "BQPB"=0)          filter("CZRQ"<:2)   --//注意Predicate Information部分,仅仅在索引中探查。

相关推荐