Oracle删除语句:级联删除后已删除多少行

3

我正在执行像这样的语句:

DELETE FROM USER WHERE USER_ID=1;

在SQLDeveloper中。

由于用户在许多表中都有引用(例如,用户有订单、设置等),因此我们激活了ON DELETE CASCADE,这样就不需要手动删除每一行。然而,在开发过程中,我们希望知道有多少行以及哪些表是通过级联删除“自动”删除的。

有没有办法找出这个问题。可以通过SQL语句,在sqldeveloper中直接从日志文件中获得,或者任何其他想法吗?


我认为这是不可能的。 - user330315
1
我不知道有这样的功能,我猜这种功能是不存在的。但是你可以递归查询数据字典(查看DBA_CONSTRAINTS),并计算详细表中受影响的行数。 - Petr Pribyl
1个回答

4
虽然使用sql%rowcount无法实现此功能,但如果编写触发器代码,则可以实现。但这意味着您需要在要监视的所有表上设置触发器。同时,触发器会稍微减缓操作速度。
例如:
SQL> select * from one;

        ID
----------
         1
         2

SQL> select * from child_of_one;

        ID       O_ID
---------- ----------
         1          1
         2          1
         3          1
         4          2
         5          2
         6          2
         7          2
         8          2

我们需要一个包规范来保存一个表和计数的数组:
SQL> create or replace package foo
  2  as
  3    type rowcount_tab is table of pls_integer index by varchar2(30);
  4    t_rowcount rowcount_tab;
  5  end foo;
  6  /

Package created.

我们希望在顶层表上设置一个触发器,将这些计数重置为零:

SQL> create or replace trigger one_biud
  2  before insert or update or delete
  3  on one
  4  declare
  5  begin
  6    foo.t_rowcount.delete;
  7  end;
  8  /

Trigger created.

假设您只对顶级表中的数组感兴趣。如果不是这样,您需要在每个表上使用触发器foo.t_rowcount.delete('TABLE_NAME')

现在,在每个有兴趣的表上设置每行后触发器来设置数组:

SQL> create or replace trigger one_aiudfer
  2  after insert or update or delete
  3  on one
  4  for each row
  5  declare
  6  begin
  7    if (foo.t_rowcount.exists('ONE'))
  8    then
  9      foo.t_rowcount('ONE') := nvl(foo.t_rowcount('ONE'), 0)+1;
 10    else
 11      foo.t_rowcount('ONE') := 1;
 12    end if;
 13  end;
 14  /

Trigger created.

SQL> create or replace trigger child_of_one_aiudfer
  2  after insert or update or delete
  3  on child_of_one
  4  for each row
  5  declare
  6  begin
  7    if (foo.t_rowcount.exists('CHILD_OF_ONE'))
  8    then
  9      foo.t_rowcount('CHILD_OF_ONE') := nvl(foo.t_rowcount('CHILD_OF_ONE'), 0)+1;
 10    else
 11      foo.t_rowcount('CHILD_OF_ONE') := 1;
 12    end if;
 13  end;
 14  /

Trigger created.

现在当我们删除或者做其他操作时:

SQL> delete from one where id = 1;

1 row deleted.

SQL> declare
  2    v_table varchar2(30);
  3  begin
  4    v_table := foo.t_rowcount.first;
  5    loop
  6       exit when v_table is null;
  7             dbms_output.put_line(v_table || ' ' || foo.t_rowcount(v_table) || ' rows');
  8             v_table := foo.t_rowcount.next(v_table);
  9     end loop;
 10  end;
 11  /
CHILD_OF_ONE 3 rows
ONE 1 rows

PL/SQL procedure successfully completed.

SQL> delete from one where id = 2;

1 row deleted.

SQL> declare
  2    v_table varchar2(30);
  3  begin
  4    v_table := foo.t_rowcount.first;
  5    loop
  6       exit when v_table is null;
  7             dbms_output.put_line(v_table || ' ' || foo.t_rowcount(v_table) || ' rows');
  8             v_table := foo.t_rowcount.next(v_table);
  9     end loop;
 10  end;
 11  /
CHILD_OF_ONE 5 rows
ONE 1 rows

令人印象深刻,但我不明白。也许可以通过遵循一些约束条件来动态完成这个任务。 - Pugzly

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接