将游标修改为建表来提高效率

0    267    1

Tags:

👉 本文共约3541个字,系统预计阅读时间或需14分钟。

目录

    记录日期: 2014-07-30 14:25:27

    题记:

    最近一个同事说一个更新语句很慢求助与我,我看了下,这类语句的优化具有典型的代表性,于是记录下来和大家共享下。

    原sql语句:

    DECLARE

    V_PARTYNO VARCHAR2(20);

    V_APP_MP VARCHAR2(20);

    V_INTIME DATE;

    V_CNT INT;

    BEGIN

    FOR CUR IN (SELECT APPLICATION_NO FROM FRAUD_SCORE) LOOP

    SELECT MAX(PARTY_NO),

    MAX(MOBILE_PHONE_NO),

    MAX(IMPORT_CPPCDM_TIME)

    INTO V_PARTYNO,

    V_APP_MP,

    V_INTIME

    FROM RISKREPT.RKO_CDM_PROCESS

    WHERE APPLICATION_NO = CUR.APPLICATION_NO;

    SELECT COUNT(1)

    INTO V_CNT

    FROM RISKREPT.RKO_CDM_PROCESS

    WHERE PARTY_NO = V_PARTYNO

    AND MONTHS_BETWEEN(V_INTIME,

    IMPORT_CPPCDM_TIME) <= 12;

    UPDATE FRAUD_SCORE

    SET APP_LOANCARD_NUM = V_CNT

    WHERE APPLICATION_NO = CUR.APPLICATION_NO;

    COMMIT;

    END LOOP;

    INSERT INTO FRAUD_SCORE_LOG

    (VAR_NAME,

    FINISH_TIME,

    SEQ_NO)

    VALUES

    ('APP_LOANCARD_NUM',

    SYSDATE,

    '8');

    END P_APP_LOANCARD_NUM;

    看下数据量:

    img

    分析:表FRAUD_SCORE 大约有5147929 行数据,采用这种方式大约需要24小时,所以优化。。。。。。

    这个块中表RISKREPT.RKO_CDM_PROCESS有多次扫描,该表也是400W的数据量,所以从这里下手优化,一般来说,建表的方式比游标要快,所以,提供三种优化方式:

    ---------------优化后的三种办法:

    -------------------------------- 第一种办法 (推荐)----- 尽量将语句简单化

    --------- 如果是存过中就用truncate + append + nologging 完成

    ------------ 我开了并行,大约10分钟,不开并行的话2个小时肯定跑完了

    *----------这种方式表建立完成后最好把表的并行模式修改为*1

    Create Table MHQ_TMP_A nologging parallel 20 As

    SELECT A.ROWID ROWIDS,

    B.APPLICATION_NO,

    MAX(PARTY_NO) PARTY_NO,

    MAX(MOBILE_PHONE_NO) MOBILE_PHONE_NO,

    MAX(IMPORT_CPPCDM_TIME) IMPORT_CPPCDM_TIME

    FROM RISKPUBFLN.FRAUD_SCORE A,

    RISKREPT.RKO_CDM_PROCESS B

    WHERE B.APPLICATION_NO = A.APPLICATION_NO

    GROUP BY A.ROWID,

    B.APPLICATION_NO ;

    Create Table MHQ_TMP_B nologging parallel 20 As

    SELECT A.ROWIDS,

    COUNT(1) COUNTS

    FROM MHQ_TMP_A A,

    RISKREPT.RKO_CDM_PROCESS B

    WHERE A.PARTY_NO = B.PARTY_NO

    AND MONTHS_BETWEEN(A.IMPORT_CPPCDM_TIME,

    B.IMPORT_CPPCDM_TIME) <= 12

    GROUP BY A.ROWIDS;

    ---select count(1) from MHQ_TMP_B;

    MERGE INTO RISKPUBFLN.FRAUD_SCORE A

    USING MHQ_TMP_B B

    本人提供Oracle(OCP、OCM)、MySQL(OCP)、PostgreSQL(PGCA、PGCE、PGCM)等数据库的培训和考证业务,私聊QQ646634621或微信dbaup66,谢谢!
    AiDBA后续精彩内容已被站长无情隐藏,请输入验证码解锁本文!
    验证码:
    获取验证码: 请先关注本站微信公众号,然后回复“验证码”,获取验证码。在微信里搜索“AiDBA”或者“dbaup6”或者微信扫描右侧二维码都可以关注本站微信公众号。

    标签:

    Avatar photo

    小麦苗

    学习或考证,均可联系麦老师,请加微信db_bao或QQ646634621

    您可能还喜欢...

    发表回复