2016-11-17 71 views
-1

我的臨時表是 -如何將動態光標傳遞到臨時表中?

begin 
    stmt := Create local temporary TABLE 

      FMO_APP.DYNAMICSQL(ENGINE_FAMILY_CODE VARCHAR2(30 BYTE), 
           CONTRACT_NAME VARCHAR2(200 BYTE) NOT NULL 
           ENABLE, 
           CONTRACT_SEQ_ID NUMBER(9, 0) NOT NULL 
           ENABLE, 
           USAGE_MONTH VARCHAR2(10 BYTE), 
           INVOICE_NUM VARCHAR2(14 BYTE) NOT NULL, 

           INVOICE_AMT NUMBER(15, 2), 
           INVOICE_BILLING_DATE DATE, 
           CREATED_DATE DATE, 
           BILLING_INVOICE_TYPE VARCHAR2(255 BYTE), 
           EFH NUMBER(11, 3), 
           EC NUMBER(9, 0), 
           CANCELLED_INVOICE_NUM VARCHAR2(14 BYTE), 
           RESTORED_ESN VARCHAR2(12 BYTE), 
           PAYMENT_TERM_TEXT VARCHAR2(60 BYTE), 
           RECON_INVOICE_NUM VARCHAR2(1 BYTE), 
           RECON_PERIOD VARCHAR2(1 BYTE), 
           PAYMENT_DUE_DATE DATE, 
           CONTRACT_CODE VARCHAR2(4 BYTE) NOT NULL 
           ENABLE, 
           PRODUCT_LINE_CODE VARCHAR2(20 BYTE) NOT NULL ENABLE, 
           PAYMENT_STATUS VARCHAR2(50 BYTE), 
           RN NUMBER, 
           COUNT NUMBER) on Commit Delete Rows; 

    execute immediate stmt; 

    insert into FMO_APP.DYNAMICSQL (cur_result); --is this correct ? 

    end; 

動態遊標是 -

open cur for v_sql; 
LOOP 
    FETCH cur 
    INTO cur_result; 
    EXIT WHEN cur%NOTFOUND; 

--v_sql is a dynamic query. 
--cur is ref_cur. 
--cur_result  dynamicsql%ROWTYPE; 

我需要從遊標使用臨時表來存儲值,並用它後來更新表。

以下是完整的代碼。

BEGIN 


usage_month_parameters := usage_month_array(); 

open cur for v_sql; 
LOOP 
    FETCH cur 
    INTO cur_result; 
    EXIT WHEN cur%NOTFOUND; 

    begin 
    stmt := Create local temporary TABLE 

      FMO_APP.DYNAMICSQL(ENGINE_FAMILY_CODE VARCHAR2(30 BYTE), 
           CONTRACT_NAME VARCHAR2(200 BYTE) NOT NULL 
           ENABLE, 
           CONTRACT_SEQ_ID NUMBER(9, 0) NOT NULL 
           ENABLE, 
           USAGE_MONTH VARCHAR2(10 BYTE), 
           INVOICE_NUM VARCHAR2(14 BYTE) NOT NULL, 

           INVOICE_AMT NUMBER(15, 2), 
           INVOICE_BILLING_DATE DATE, 
           CREATED_DATE DATE, 
           BILLING_INVOICE_TYPE VARCHAR2(255 BYTE), 
           EFH NUMBER(11, 3), 
           EC NUMBER(9, 0), 
           CANCELLED_INVOICE_NUM VARCHAR2(14 BYTE), 
           RESTORED_ESN VARCHAR2(12 BYTE), 
           PAYMENT_TERM_TEXT VARCHAR2(60 BYTE), 
           RECON_INVOICE_NUM VARCHAR2(1 BYTE), 
           RECON_PERIOD VARCHAR2(1 BYTE), 
           PAYMENT_DUE_DATE DATE, 
           CONTRACT_CODE VARCHAR2(4 BYTE) NOT NULL 
           ENABLE, 
           PRODUCT_LINE_CODE VARCHAR2(20 BYTE) NOT NULL ENABLE, 
           PAYMENT_STATUS VARCHAR2(50 BYTE), 
           RN NUMBER, 
           COUNT NUMBER) on Commit Delete Rows; 

    execute immediate stmt; 

    insert into FMO_APP.DYNAMICSQL (cur_result); 

    end; 

    begin 
    select fi.billing_invoice_type, mi.tag_type 
     into v_inv_type, v_tag_type 
     from fmo_op2_invoice fi, fmo_op2_manual_invoice_items mi 
    where fi.invoice_num = mi.invoice_num; 

    if upper(v_inv_type) = 'M' and upper(v_tag_type) = 'P' then 

     select count(distinct mi.item_date) 
     into item_date_count 
     from fmo_op2_manual_invoice_items mi 
     where mi.invoice_num = cur_result.INVOICE_NUM; 

     if item_date_count = 1 then 
     select distinct mi.item_date 
      into v_item_date 
      from fmo_op2_manual_invoice_items mi 
     where mi.invoice_num = cur_result.invoice_num; 

     SELECT to_char(v_item_date, 'yyyy - MM - dd') 
      into var_usage_month 
      from dual; 

     elsif item_date_count > 1 then 
     var_usage_month := to_char('MULTIPLE'); 

     else 
     var_usage_month := to_char(cur_result.USAGE_MONTH, 
            'yyyy - MM - dd'); 
     end if; 

     BEGIN 
     v_usage_month_arr.EXTEND; 

     v_usage_month_arr(var_num) := usage_month_value_obj(cur_result.ENGINE_FAMILY_CODE, 

                  cur_result.CONTRACT_NAME, 
                  cur_result.CONTRACT_SEQ_ID, 
                  var_usage_month, 
                  cur_result.INVOICE_NUM, 
                  cur_result.INVOICE_AMT, 
                  cur_result.INVOICE_BILLING_DATE, 
                  cur_result.CREATED_DATE, 
                  cur_result.BILLING_INVOICE_TYPE, 
                  cur_result.EFH, 
                  cur_result.EC, 
                  cur_result.CANCELLED_INVOICE_NUM, 
                  cur_result.RESTORED_ESN, 
                  cur_result.PAYMENT_TERM_TEXT, 
                  cur_result.RECON_INVOICE_NUM, 
                  cur_result.RECON_PERIOD, 
                  cur_result.PAYMENT_DUE_DATE, 
                  cur_result.CONTRACT_CODE, 
                  cur_result.PRODUCT_LINE_CODE, 
                  cur_result.PAYMENT_STATUS 

                  ); 
     end; 

    else 
     var_usage_month := to_char(cur_result.USAGE_MONTH, 
           'yyyy - MM - dd'); 

    end if; 
    end; 

end loop; 
close cur; 

OPEN p_out_contract_data FOR 
    select cast(v_usage_month_arr as fmo_op2_manual_table_type) 
    from dual; 

commit; 
+1

你爲什麼試圖在代碼中創建表?除了您的語法不正確以及缺少將您的create table語句轉換爲字符串的單引號(因此您可以將它存儲在stmt變量中)之外,它看起來像表具有固定的結構。 我會做的是在數據庫中創建表作爲GTT(全局臨時表 - 表是永久對象,但數據僅存儲在會話級別),然後在代碼中引用它。但是,您的代碼並不實際引用您的臨時表;所以爲什麼要打擾呢? – Boneist

回答

1

您對Oracle臨時表的理解是完全錯誤的。您需要創建它只有一次,任何存儲的代碼外(過程,函數,包):

create temporary table my_temporary_table (...) on commit delete rows; 

之後,你就可以在任何地方使用它,你需要的。與數據使用insert ... select語句填充:

procedure my_proc (...) is 
    ... 
begin 
    insert into my_temporary_table (...) 
    select ... 
    from source_table; 

end; 

表將在提交之後被清除。