使用hint来调优sql语句(72天)

最近生产发现有一个sql语句运行耗时达5000多秒。 抓出来sql_id一看,sql倒不是一个很长的语句。结构也很简单。如下。

select company_code, sap_company_id
  from data_company_code
 where company_code not in
       (SELECT distinct l9_company_code
          FROM detailed_data_info_v a, refund_request b
         WHERE a.financial_activity = 'RFND'
           and a.refund_method = 'AP'
           AND a.refund_id = b.refund_id
           AND b.refund_status = 'P'
           AND b. REVERSAL_TRANS_ID is null
           AND posting_date = TO_DATE(20140511, 'YYYYMMDD'))

执行计划如下:

Execution Plan                                                                                                           
-----------------------------------------------------------------------------------------------------------------------  
| Id  | Operation                     | Name                  | Rows  | Bytes | Cost (%CPU)| Time     | Pstart| Pstop |  
-----------------------------------------------------------------------------------------------------------------------  
|   0 | SELECT STATEMENT              |                       |       |       |  2696K(100)|          |       |       |  
|   1 |  FILTER                       |                       |       |       |            |          |       |       |  
|   2 |   MAT_VIEW ACCESS FULL        | DATA_COMPANY_CODE     |     5 |    35 |     3   (0)| 00:00:01 |       |       |  
|   3 |   NESTED LOOPS                |                       |       |       |            |          |       |       |  
|   4 |    NESTED LOOPS               |                       |     1 |    68 |  1078K  (2)| 03:35:43 |       |       |  
|   5 |     PARTITION RANGE ALL       |                       |     1 |    59 |  1078K  (2)| 03:35:43 |     1 |   366 |  
|   6 |      TABLE ACCESS FULL        | DETAILED_DATA         |     1 |    59 |  1078K  (2)| 03:35:43 |     1 |   366 |  
|   7 |     INDEX UNIQUE SCAN         | REFUND_REQUEST_PK     |     1 |       |     1   (0)| 00:00:01 |       |       |  
|   8 |    TABLE ACCESS BY INDEX ROWID| REFUND_REQUEST        |     1 |     9 |     1   (0)| 00:00:01 |       |       |  
-----------------------------------------------------------------------------------------------------------------------  

查看最后的输出表data_company_code,发现是一个数据字典表,里面的数据很少。只有5条。 表 detailed_data_info_v是一个视图,里面参照的基表只有1个. detailed_data_info 它是一个历史表,里面有3亿多条数据。而且对应的主键在查询条件中也没有,这也是这个sql执行慢的主要原因。 表 REFUND_REQUEST 是一个应用表,里面的数据就几百条。 明白了大概的情况之后。 首先从视图下手。看看 a.refund_method , a.refund_id,company_code都运用了大量的decode,可以看到都是基于financial_activity来做的过滤,所以直接可以提出其他的条件过滤,直接使用基表来去得所需的条件。

DECODE( FINANCIAL_ACTIVITY,
'RFND',
 DATA_FIELD_1) as REFUND_ID 

DECODE( FINANCIAL_ACTIVITY,
'RFND',
 DATA_FIELD_3) as REFUND_METHOD

DECODE( FINANCIAL_ACTIVITY,
'RFND',
 DATA_FIELD_10 ,
 'WERRE',
 DATA_FIELD_10 ,
。。。。。。。
 DATA_FIELD_19 ,
 'BCK',
 DATA_FIELD_20 ,
 'DSPCAN',
 DATA_FIELD_7 ,
 'DSPREJ',
 DATA_FIELD_7 ,
 'WER',
 DATA_FIELD_7 ,
 'WWER',
 DATA_FIELD_7 ,
 'DD',
 DATA_FIELD_7 ,
 'reer',
 DATA_FIELD_7 ,
 'tttt',
 DATA_FIELD_8 ,
 'xxxx',
 DATA_FIELD_9) as COMPANY_CODE 

所以把查询可以构造成几个子查询。黄色是做了变化的部分。

select a.DATA_FIELD_10 l9_company_code
           from DETAILED_DATA a
           where a.financial_activity = 'RFND'
           and a.DATA_FIELD_3 = 'AP'
           and a.posting_date = TO_DATE(20140511, 'YYYYMMDD')

另外一个子查询。

select refund_id
               from ar1_refund_request b
               where b.refund_status='P'
           and b.reversal_trans_id is null  
            /
  no rows selected

结果已查询,让我大跌眼镜,竟然没有匹配的值。但是sql语句还是会不断的去做无用功。查了半天,结果返回了一个Null。 找到了基本的方向,如果查询条件中没有匹配的值,至少可以不用再从3亿多条记录的表里去全表扫描了。 在测试下面的查询时,如果屏蔽掉条件a.financial_activity = 'RFND',查询就会直接先进入refund_request了。

SQL> select 
            distinct a.DATA_FIELD_10 l9_company_code  
              from DETAILED_DATA a
         where a.DATA_FIELD_3 = 'AP'
           --and a.financial_activity = 'RFND'
           and a.posting_date = TO_DATE(20140512, 'YYYYMMDD')
           and exists (select 1
                  from refund_request b
                 where b.refund_id = a.DATA_FIELD_1
                   and b.refund_status = 'P'
                   and b.reversal_trans_id is null) 
   /
no rows selected
Elapsed: 00:00:00.01

如果没有数据,马上就返回了。类似于这样的方式 select xxx from huge_table where 1!=1 但是已加入条件financial_activity就开始扫描大表,看来只能使用Hint来强制指定表的访问顺序了。当然了使用hint也是玩不得以而为之。不建议一开始调就考虑hint.

SQL> select company_code, sap_company_id
      from ar9_company_code
     where company_code not in
       (select /*+leading(b,a)*/
        distinct a.DATA_FIELD_10 l9_company_code --,financial_activity,DATA_FIELD_3,posting_date 
          from AR1_GL_DETAILED_DATA a
         where a.DATA_FIELD_3 = 'AP'
           and a.financial_activity = 'RFND'
           and a.posting_date = TO_DATE(20140512, 'YYYYMMDD')
           and exists (select 1
                  from ar1_refund_request b
                 where b.refund_id = a.DATA_FIELD_1
                   and b.refund_status = 'P'
                   and b.reversal_trans_id is null))
SQL> /
COMPAN SAP_COMPANY_ID
------ --------------
AE                1010
XX                1068
XXE               1067
DS                1027
EER               1019
Elapsed: 00:00:00.01

原文发布于微信公众号 - 杨建荣的学习笔记(jianrong-notes)

原文发表时间:2014-05-14

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

来自专栏james大数据架构

通用分页存储过程

/*通用分页存储过程*/ USE HotelManagementSystem GO IF EXISTS(SELECT * FROM sys.objects WH...

25980
来自专栏杨建荣的学习笔记

关于索引扫描的极速调优实战(第一篇) (r3笔记第81天)

一般在生产环境中,如果某个查询中涉及一个大表,走索引扫描是显然是最值得推荐的方式,但是索引扫描有unique index scan, range scan,sk...

25740
来自专栏Jerry的SAP技术分享

使用ABAP(ADBC)和Java(JDBC)连接SAP HANA数据库

在表DBCON里维护一条记录,指向HANA数据库。con_ENV里填入HANA数据库的主机名和端口号。如vmXXXX:30015

44730
来自专栏乐沙弥的世界

Oracle 性能相关常用脚本(SQL)

在缺乏的可视化工具来监控数据库性能的情形下,常用的脚本就派上用场了,下面提供几个关于Oracle性能相关的脚本供大家参考。以下脚本均在Oracle 10g测试通...

14220
来自专栏杨建荣的学习笔记

关于物化视图疑问(32天)

--初始化操作, 创建两个用户一个,testo,一个test. 在testo上创建表,test上创建物化视图。 SQL> create user testo ...

36950
来自专栏数据库新发现

Oracle9i新特性-使用DBMS_METADATA包获得对象DDL语句

从Oracle9i开始Oracle提供了一个新的系统包DBMS_METADATA,可以用于提取对象创建的DDL语句。

15220
来自专栏乐沙弥的世界

避免游标多次遍历

游标在数据库领域被广泛使用,尤其是对于需要将SQL语句返回的数据集进行逐行处理的时候。这为数据处理提供了极大的便利性,然游标的不当 使用从某种程度...

8420
来自专栏WindCoder

where in与join 查询

Oracle:当前所用版本中,限制in中的参数不能超过 1000个。当超出时会被报错"ORA-01795异常(where in超过1000)的解决"。

47200
来自专栏乐沙弥的世界

SQL 基础--> 视图(CREATE VIEW)

视图来源于表,所有对视图数据的修改最终都会被反映到视图的基表中,这些修改必须服从基表的完整性约束,并同样会触发定义

8030
来自专栏Java学习123

ORACLE数据库日常维护

63780

扫码关注云+社区

领取腾讯云代金券