1

获取和输出在下面的 FORALL 语句中插入了多少行的最佳方法是什么。我已经看到了 SQL%BULK_ROWCOUNT 但我不确定它在下面的语句中是如何工作的。

是吗

DBMS_OUTPUT.('rows inserted '||SQL%BULK_ROWCOUNT||'');

以上内容是否需要进入另一个 FORALL 语句?对于下面的代码,我将如何实现这一点?

DECLARE
  TYPE t_arc_act_plus_trigger1 IS TABLE OF arc_act_plus_triggers1%ROWTYPE;
  v_arc_act_plus_triggers1 t_arc_act_plus_trigger1;

  CURSOR c_arc_act_plus_triggers1 IS
      SELECT /*+ PARALLEL */ apt.*
        FROM act_plus_triggers1 apt
       WHERE NOT EXISTS
                (SELECT 1
                   FROM act_plus_triggers_copy1 aptc
                  WHERE aptc.surr_id = apt.surr_id)
         AND apt.status IN ('EXT', 'EXP');
BEGIN

    OPEN c_arc_act_plus_triggers1;
    LOOP
        FETCH c_arc_act_plus_triggers1 BULK COLLECT INTO v_arc_act_plus_triggers1 LIMIT 10000;  -- limit to 10k to avoid out of memory

        FORALL i IN 1..v_arc_act_plus_triggers1.COUNT
        INSERT /*+ APPEND_VALUES */ INTO arc_act_plus_triggers1 values v_arc_act_plus_triggers1(i);  


        Com0932.get_parameter ('ACT_ARCHIVE_TRIGGER_STOP_YN',l_STOP_PROGRAM_YN);         
        IF l_STOP_PROGRAM_YN = 'Y' THEN
            p_location('insert_into_arc_act_plus - STOP_PROGRAM_YN flag = '||l_STOP_PROGRAM_YN||' so ROLLBACK'); 
            ROLLBACK;
            EXIT;
        END IF; 

        -- **************************************************
        -- Output how many records have been inserted here???
        -- **************************************************

        -- commit after every 10000 records into arc_act_plus_triggers1   
        COMMIT;     
        EXIT WHEN c_arc_act_plus_triggers1%NOTFOUND;


    END LOOP;
    CLOSE c_arc_act_plus_triggers1;

END;
4

1 回答 1

0

我没有检查,因为我没有什么要测试的,所以请原谅任何“缺少分号类型错误”,恐怕我无法进行性能检查。

您的代码似乎根据存档中不存在的情况选择要插入存档表的行。因此,只需使用基于 SELECT 的 INSERT,该 SELECT 受合适的 ROWNUM 值限制。一旦你提交,那么下一次循环它不会尝试获取已经归档的行,因为你刚刚提交它们。

我认为这应该与使用其更简单的优势——Occams Razor 和所有这些来扩大插入物一样快,甚至更快。

DECLARE
  l_commit_count NUMBER := 10000;
  l_rows_copied NUMBER := 0;
BEGIN
    DBMS_OUTPUT.PUT_LINE('Started at '||TO_DATE(SYSDATE, 'DD_MON_YYY HH24:MI:SS');
    LOOP
        INSERT /*+APPEND */
        INTO c_arc_act_plus_triggers1
        SELECT /*+ PARALLEL */ apt.*
            FROM act_plus_triggers1 apt
           WHERE NOT EXISTS
                    (SELECT 1
                       FROM act_plus_triggers_copy1 aptc
                      WHERE aptc.surr_id = apt.surr_id)
             AND apt.status IN ('EXT', 'EXP')
             AND rownum < l_commit_count;

        COMMIT;         
        l_rows := l_rows + SQL%ROWCOUNT;
        EXIT WHEN SQL%ROWCOUNT < 1;
    END LOOP        
    DBMS_OUTPUT.PUT_LINE('Finished at '||TO_DATE(SYSDATE, 'DD_MON_YYY HH24:MI:SS');
    DBMS_OUTPUT.PUT_LINE(TO_CHAR(l_rows)||' rows copied to the archive table');
END;
于 2017-02-08T16:02:57.983 回答