我尝试与SYS_REFCURSOR的集合结果进行比较:
declare
v_RC sys_refcursor;
v_RC_union sys_refcursor;
v_REC userA.table1%rowtype;
v_REC_union userB.table2%rowtype;
i number := 0;
j number := 0;
z number := 0;
begin
open v_RC for select * from userA.table1;
open v_RC_union for select * from userB.table2;
loop fetch v_RC into v_REC;
exit when v_RC%notfound;
i := i+1;
loop fetch v_RC_union into v_REC_union;--
exit when v_RC_union%notfound;
j := j+1;
If v_REC_union.id= v_REC.id then
z :=z+1;
End if;
end loop;
end loop;
close v_RC;
close v_RC_union;
dbms_output.put_line(z);--too small
dbms_output.put_line('v_RC: '||i||', v_REC_union: '||j);
end;
我知道两个光标的行数都正确(i和j都可以),但是相等行(z)的结果是错误的(太小)。
答案 0 :(得分:1)
在外循环的第一次迭代中,您正在消耗v_RC_union
中的所有行。您似乎希望每次循环时都可以看到所有这些行,但是第二个ref光标不会重置为开始(并且不可能)。
如果将调试添加到代码中,则可以看到发生的情况;我创建了两个小的虚拟表,每个虚拟表中有三个匹配ID和一个不匹配ID:
create table table1 (id, dummy) as select level + 1, 'x' from dual connect by level <= 4;
create table table2 (id, dummy) as select level, 'x' from dual connect by level <= 4;
declare
...
begin
open v_RC for select * from table1;
open v_RC_union for select * from table2;
loop
fetch v_RC into v_REC;
exit when v_RC%notfound;
i := i+1;
dbms_output.put_line('i: ' || i || ' table1 id ' || v_REC.id);
loop
fetch v_RC_union into v_REC_union;--
exit when v_RC_union%notfound;
j := j+1;
dbms_output.put_line('i: ' || i || ' j: ' || j || ' table1 id ' || v_REC_union.id);
If v_REC_union.id= v_REC.id then
z :=z+1;
end if;
end loop;
end loop;
close v_RC;
close v_RC_union;
dbms_output.put_line('z: ' || z);--too small
dbms_output.put_line('v_RC: '||i||', v_REC_union: '||j);
end;
/
输出是:
i: 1 table1 id 2
i: 1 j: 1 table1 id 1
i: 1 j: 2 table1 id 2
i: 1 j: 3 table1 id 3
i: 1 j: 4 table1 id 4
i: 2 table1 id 3
i: 3 table1 id 4
i: 4 table1 id 5
z: 1
v_RC: 4, v_REC_union: 4
在第一次迭代中,当i
为1时,您将执行内部循环并从v_RC_union
中获取所有行,仅在notfound
时停止。假设其中一个与第一行v_RC
的ID匹配,则z
递增,因此在第一个外部循环之后为1或0。
在第二次迭代中,当i
为2时,由于您已经用尽v_RC_union
结果集,因此内部循环将在第一次获取后立即退出。您在第一次迭代中获取了所有行,因此没有任何可获取的内容。因此,没有第二行v_RC
的ID不会与任何东西进行比较,并且z
不会被触及,因为它没有那么远。
对于外部循环的所有其他迭代,依此类推。内部循环始终会立即退出,并且不会做任何有用的事情。
您可以在答案中每次显示时都重新查询table2
以获得特定的ID,尽管如果仅对它们进行计数而不使用任何列值,则实际上并不需要打开/循环/ fetch并可以将该查询更改为一个标量变量。即使对于逐行处理,这似乎也不是很有效。
如果要坚持使用全表查询,可以使用集合:
declare
type t_table1 is table of table1%rowtype;
type t_table2 is table of table2%rowtype;
v_table1 t_table1;
v_table2 t_table2;
i number := 0;
j number := 0;
z number := 0;
begin
select * bulk collect into v_table1 from table1;
select * bulk collect into v_table2 from table2;
for r1 in v_table1.first..v_table1.last loop
i := i+1;
dbms_output.put_line('i: ' || i || ' table1 id ' || v_table1(r1).id);
j := 0;
for r2 in v_table2.first..v_table2.last loop
j := j+1;
dbms_output.put_line('i: ' || i || ' j: ' || j || ' table2 id ' || v_table2(r2).id);
if v_table2(r2).id = v_table1(r1).id then
z := z+1;
end if;
end loop;
end loop;
dbms_output.put_line('z: ' || z);
dbms_output.put_line('v_RC: '||i||', v_REC_union: '||j);
end;
/
通过遍历两个集合,每次i
迭代都会看到每个j
行,并且可以比较ID。
i: 1 table1 id 2
i: 1 j: 1 table2 id 1
i: 1 j: 2 table2 id 2
i: 1 j: 3 table2 id 3
i: 1 j: 4 table2 id 4
i: 2 table1 id 3
i: 2 j: 1 table2 id 1
i: 2 j: 2 table2 id 2
i: 2 j: 3 table2 id 3
i: 2 j: 4 table2 id 4
i: 3 table1 id 4
i: 3 j: 1 table2 id 1
i: 3 j: 2 table2 id 2
i: 3 j: 3 table2 id 3
i: 3 j: 4 table2 id 4
i: 4 table1 id 5
i: 4 j: 1 table2 id 1
i: 4 j: 2 table2 id 2
i: 4 j: 3 table2 id 3
i: 4 j: 4 table2 id 4
z: 3
v_RC: 4, v_REC_union: 4
当您可以使用集合运算符直接在普通SQL中进行计数和比较时,这似乎仍然是很多工作,例如:
select
(select count(*) from table1) as i,
(select count(*) from table2) as j,
(select count(*) from (select id from table1 intersect select id from table2)) as z
from dual;
I J Z
---------- ---------- ----------
4 4 3
在这里使用PL / SQL似乎并没有太大的好处。 SQL版本可能会执行更多的工作,因为它两次查询两个表(尽管可能会命中缓存),但是无论如何,您都可以通过CTE避免这种情况,并且总体上可能比使用PL / SQL更快。
答案 1 :(得分:0)
我更改了顺序,现在可以正常工作
open v_RC for select * from select * from userA.table1;
loop fetch v_RC into v_REC;
exit when v_RC%notfound;
open v_RC_union for select * from select * from userB.table2 where id = v_REC.id;
loop fetch v_RC_union into v_REC_union;
exit when v_RC_union%notfound;
if v_REC.id = v_REC_union.id then
z :=z+1;
dbms_output.put_line(v_REC.id ||'='|| v_REC_union.id);
end if;
end loop;
close v_RC_union;
end loop;
close v_RC;
但我仍然不知道出了什么问题?