Это похоже на некоторую ошибку или ограничение при попытке выполнить массовое связывание при возврате collection
или sys_refcursor
из другой процедуры. Тем не менее, у вас есть много других вариантов, а именно массовое связывание (циклическое использование FORALL
), которые одинаково производительны.
Альтернатива, как показано ниже:
Подготовка таблицы:
Create table employees (
first_name varchar2(100),
last_name varchar2(100),
salary number
);
Insert into employees values('A','B',1000);
Insert into employees values('C','D',2000);
Insert into employees values('E','F',3000);
Create table t1 (
first_name varchar2(100),
last_name varchar2(100)
);
--Using Objects in place of records and sys_refcursor.
Create or replace type return_value is Object (
first_name varchar2(100),
last_name varchar2(100)
);
Create or replace type v_ret_val is table of return_value;
Вариант 1: объявить P1
как Function
, а затем использовать коллекцию как прямую вставку пути.
CREATE OR REPLACE function p1 (v_salary NUMBER)
return v_ret_val
AS
abc v_ret_val;
BEGIN
SELECT return_value(first_name,last_name)
Bulk collect into
abc
FROM employees
WHERE salary > v_salary;
Return abc;
END;
---------------
CREATE OR REPLACE PROCEDURE p2 (
v_sal NUMBER
) AS
k v_ret_val;
BEGIN
k:=p1(v_sal);
--Displaying values of collection
For i in 1..k.count
Loop
dbms_output.put_line(k(i).first_name || k(i).last_name);
End Loop;
--Direct path insert
INSERT /*+Append*/ INTO t1
Select t.first_name,t.last_name
from (table( k )) t;
COMMIT;
END;
Вариант 2: объявить P1
как procedure
с параметром OUT
и затем использовать коллекцию как прямой путь вставки.
CREATE OR REPLACE procedure p11 (
v_salary NUMBER
,output_value OUT v_ret_val
)
AS
BEGIN
SELECT return_value(first_name,last_name)
Bulk collect into
output_value
FROM employees
WHERE salary > v_salary;
END;
------------------------------
CREATE OR REPLACE PROCEDURE p2 ( v_sal NUMBER)
AS
k v_ret_val;
BEGIN
p11(v_sal,k);
For i in 1..k.count
Loop
dbms_output.put_line(k(i).first_name || k(i).last_name);
End Loop;
INSERT /*+Append*/ INTO t1
Select t.first_name,t.last_name
from (table( k )) t;
END;
Вариант 3: объявление P1
как procedure
с параметром OUT
и затем Merging
коллекции с целевой таблицей.
CREATE OR REPLACE PROCEDURE p2 (
v_sal NUMBER
) AS
k v_ret_val;
BEGIN
p11(v_sal,k);
For i in 1..k.count
Loop
dbms_output.put_line(k(i).first_name || k(i).last_name);
End Loop;
MERGE INTO t1 tgt
Using ( Select t.first_name,
t.last_name
from table ( k ) t
) src
on (
tgt.first_name = src.first_name
and tgt.last_name = src.first_name
)
When not matched
then
insert (
tgt.first_name,
tgt.last_name
)
values (
src.first_name,
src.last_name
);
END;
Выполнение:
Exec P2(1000);
Выход:
SQL> Select * from t1;
FIRST_NAME LAST_NAME
-------------------------------------------
E F
C D