相同的查询,不同的性能

时间:2016-11-16 11:58:19

标签: spring oracle mybatis difference hint

有一个查询需要两分多钟。该查询的结果是960000行。 所以我使用提示。然后只需30秒。所以我在mybatis中应用查询并运行应用程序。但该查询需要超过两分钟。所以我在日志中复制查询并粘贴本地开发人员。并执行查询。但它只需要30秒。 我不知道........

环境:春天,mybatis,oracle

SELECT * 
FROM (
    SELECT /*+ USE_HASH(a b c) */ ROW_NUMBER() OVER(ORDER BY A.TRANS_DT DESC, A.TRANS_TM DESC) AS RNUM                    
          ,A.PAY_METHOD
          ,B.BRAND_NM  
          ,A.I_MID                                
          ,C.TRANS_DT  
          ,C.TRANS_TM  
          ,C.TRANS_DT || C.TRANS_TM AS TRANS_DTM   
          ,C.VACCT_VALID_DT                        
          ,C.VACCT_VALID_TM                        
          ,C.VACCT_VALID_DT||C.VACCT_VALID_TM AS VACCT_VALID_DTM 
          ,NVL(C.DEPOSIT_DT, ' ') DEPOSIT_DT                     
          ,NVL(C.DEPOSIT_TM, ' ') DEPOSIT_TM                     
          ,NVL(C.DEPOSIT_DT, ' ')||NVL(C.DEPOSIT_TM, ' ') AS DEPOSIT_DTM    
          ,NVL(C.DEPOSIT_AMT, 0) AS AMT                                     
          ,NVL(A.AMT, 0) AS INPUT_AMT                                       
          ,A.BANK_CD                                                        
          ,IONPAY.UF_GET_BANK_NAME(A.BANK_CD) AS BANK_CD_NM                 
          ,C.VACCT_NO                                                  
          ,A.BILLING_NM                                           
          ,A.REFERENCE_NO                                    
          ,A.TXID                                        
          ,A.STATUS         
          ,NVL(A.STATUS, ' ') AS INPUT_STATUS    
          ,IONPAY.UF_GET_TRANS_VACCT_STATUS_NAME(C.STATUS) AS INPUT_STATUS_NAME
          ,C.STATUS AS TRANS_STATUS  
          ,IONPAY.UF_GET_TRANS_VACCT_STA_NAME_2(C.STATUS, C.MATCH_CL, C.VACCT_VALID_DT, C.VACCT_VALID_TM) AS STATUS_NAME 
          ,A.ACQU_STATUS                 
          ,A.CANCEL_DT||A.CANCEL_TM AS REVERSAL_DATE 
          ,NVL((SELECT DESC2
                    FROM TB_CODE
                    WHERE CODE_CL = 'CHNL'
                      AND CODE1 = C.BANK_CD
                      AND CODE2 = C.CHANNEL_TYPE), ' ') AS channel
    FROM TB_TRANS_HISTORY A, TB_BO_MER_MGMT B, TB_VACCT_TRANS C
    WHERE A.I_MID = B.I_MID
      AND A.TXID = C.TXID
      AND A.I_MID = C.I_MID
      AND B.I_MID = C.I_MID
      AND A.PAY_METHOD IN ('02')
      AND A.TRANS_DT BETWEEN '20161016' AND '20161116' 
      AND A.TRANS_DT||A.TRANS_TM BETWEEN '2016101600%3A0000' AND '2016111624%3A0000'     
      AND B.TAX_NO != 'NICEPAY' 
      AND C.SIMULATION_FLG = '0' 
      AND C.STATUS IN ('0', '1', '2', '3', '4') 
) TBL
WHERE RNUM BETWEEN 210000 AND 220000

1 个答案:

答案 0 :(得分:2)

我已经遇到过这样的情况:时间差异不是在SQL查询执行中,而是在获取结果时。

Oracle JDBC驱动程序的默认提取大小为10,这意味着ResultSet以10行10行的形式提供,当有100万条记录时,这会对DB进行大量往返。必须增加提取大小。

使用PostgreSql,默认的提取大小是无限的:ResultSet被提供整个结果(或直到OutOfMemory)。可能需要降低获取大小。

在Mybatis中指定获取大小值:

在XML select语句中使用 fetchSize 属性:

<select id="listItems" fetchSize="300">SELECT ...</select>

或者在注释中使用 fetchSize 选项:

@Select("SELECT ...")
@Options(fetchSize=300)

200-500范围内的值通常是一个很好的折衷方案。