使用hint来调优sql语句

最近生产发现有一个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

时间: 2024-10-01 21:49:15

使用hint来调优sql语句的相关文章

调优SQL过程

您想学习调优 SQL PL 的一些技巧吗?如果是这样的话,本文将描述一些调优 IBM DB2 Universal Database(UDB)V8.2 for Linux,UNIX, and Windows 中的 SQL 过程的常见例子,并将特别关注从其他数据 库管理系统中移植过来的过程.动机 SQL 过程语言(SQL PL)是许多 DB2 开发人员工具箱中的一个流行工具.当将应用程序 从 Oracle.Sybase 或 Microsoft SQL 服务器迁移或移植到 DB2 中时,更是如此.然而

我如何调优SQL Server查询

原文:我如何调优SQL Server查询 我是个懒人,我只想干尽可能少的活.当我干活的时候我不想太多.是,你没看错,这看起来很糟糕,作为一个DBA这很不合格.但在今天的文章里,我想给你展示下,当你想对特定查询创建索引设计时,你如何把你的工作和思考过程传达给查询优化器.听起来很有意思?嗯,那就进入我的索引调优世界吧! 有问题的查询 我们来看下列查询: 1 DECLARE @i INT = 999 2 SELECT 3 SalesOrderID, 4 SalesOrderDetailID, 5 Ca

如何调优SQL Server查询_MsSql

在今天的文章里,我想给你展示下,当你想对特定查询创建索引设计时,如何把你的工作和思考过程传达给查询优化器.下面就一起来探讨一下吧! 有问题的查询我们来看下列查询: DECLARE @i INT = 999 SELECT SalesOrderID, SalesOrderDetailID, CarrierTrackingNumber, OrderQty, LineTotal FROM Sales.SalesOrderDetail WHERE ProductID < @i ORDER BY Carri

对hint的调优

实际工作中经常遇到开发人员加hint为提高数据的批处理的速度,但为了提高处理速度经常遇到并行的hint随意使用,并行不是万能的,不合理的使用只能阻碍运行速度,使用如下SQL说明并行问题 SELECT  /*+ LEADING(T1) USE_HASH(T_LGIN) PARALLEL(T1,8) */                              T1.RPO_NO                     ,T_LGIN.LGIN_DT                     ,T_

如何调优SQL Server查询

在今天的文章里,我想给你展示下,当你想对特定查询创建索引设计时,如何把你的工作和思考过程传达给查询优化器.下面就一起来探讨一下吧! 有问题的查询 我们来看下列查询: DECLARE @i INT = 999 SELECT SalesOrderID, SalesOrderDetailID, CarrierTrackingNumber, OrderQty, LineTotal FROM Sales.SalesOrderDetail WHERE ProductID < @i ORDER BY Carr

调优IBM DB2 UDB SQL存取路径

简介 Visual Explain 是 IBM DB2 Universal Database 中的杰出工具,程序员和 DBA 用它来详细说明 DB2 优化器为 SQL 语句所选择的存取路径.事实上,Explain 应该是您性能监控策略的 关键组件.Explain 为解决许多类型的性能问题提供了价值无法估量的信息,因为它提供这样的细节: DB2 在"幕后"所做的工作,以实现 SQL 请求的数据需求 DB2 是否使用可用的索引,如果使用,DB2 如何使用它们 为满足连接条件而访问 DB2

一次SQL Server调优经历

  前段时间数据库健康检查发现SQL Server服务器的idle时间变少,IO还是比较空闲,估计是遇到了高CPU占用的语句了. 介绍一下背景,我们公司负责运维N多的应有系统,负责提供良好的软.硬件环境,至于应用的开发质量,我们就无能为力了 解决这个问题,我的思路是: 找出CPU占用最大的语句. 分析查询计划. 优化. 1.找出语句 使用SQL Server自带的性能报表(不是报表服务),找出CPU占用最大的语句.如图1所示 图1 性能报表 我选取了"性能-按总CPU时间排在前面的查询"

一条delete语句的调优

今天刚上上班,就接到客户的邮件,说生产环境中执行某一条delete sql语句的时间超过了3个小时.最后客户无奈取消了这次数据清理,准备今天在申请时间重做.所以希望我在下午之前能够调优一下sql语句. 我拿到sql语句.是一个简单的delete语句,这个表是一个分区表,表中的数据大约有6亿条,要删除的数据大概有900多万条. delete event           where cycle_code = 25           and cycle_month = 2           a

SQL Server调优系列玩转篇(如何利用查询提示(Hint)引导语句运行)

原文:SQL Server调优系列玩转篇(如何利用查询提示(Hint)引导语句运行) 前言 前面几篇我们分析了关于SQL Server关于性能调优的一系列内容,我把它分为两个模块. 第一个模块注重基础内容的掌握,共分6篇文章完成,内容涵盖一系列基础运算算法,详细分析了如何查看执行计划.掌握执行计划优化点,并一一列举了日常我们平常所写的T-SQL语句所会应用的运算符.我相信你平常所写的T-SQL语句在这几篇文章中都能找到相应的分解运算符. 第二个模块注重SQL Server执行T-SQL语句的时候