Oracle级联删除

Posted

技术标签:

【中文标题】Oracle级联删除【英文标题】:Oracle cascade delete 【发布时间】:2011-12-20 06:43:58 【问题描述】:

表上的级联删除是否比单个删除语句(在单个 plsql 块中执行)更有效?

【问题讨论】:

【参考方案1】:

cascade delete 所做的是发出单独的删除语句。

检查以下测试用例:

create table parent 
 (parent_id number, 
  parent_name varchar2(30), 
  constraint parent_pk primary key (parent_id) using index);

create table child 
 (child_id   number,
  parent_id  number,
  child_name varchar2(30),
  constraint child_pk primary key (parent_id, child_id) using index,
  constraint child_fk01 foreign key (parent_id) 
    references parent (parent_id) 
      on delete cascade;
 );


insert into parent
 (parent_id, parent_name)
select object_id, object_name from dba_objects where rownum <= 10000;

begin
  for i in 1..10
  loop
    insert into child
      (child_id, parent_id, child_name)
    select i, parent_id, parent_name
      from parent;
  end loop;
end;
/

exec dbms_stats.gather_table_stats (tabname => 'PARENT', cascade => true);
exec dbms_stats.gather_table_stats (tabname => 'CHILD', cascade => true);

exec dbms_monitor.session_trace_enable;
alter table child drop constraint child_fk01;
alter table child add constraint child_fk01 foreign key (parent_id)
  references parent (parent_id) on delete cascade enable novalidate ;
delete from parent;
rollback;

在跟踪文件中,你会发现这样一行:

  delete from "<MY_SCHEMA_NAME>"."CHILD" where "PARENT_ID" = :1
END OF STMT
PARSE #6:c=0,e=182,p=0,cr=0,cu=0,mis=1,r=0,dep=1,og=4,tim=1293353992514766
EXEC #6:c=0,e=545,p=0,cr=2,cu=32,mis=1,r=10,dep=1,og=4,tim=1293353992515354
EXEC #6:c=0,e=233,p=0,cr=2,cu=30,mis=0,r=10,dep=1,og=4,tim=1293353992515644
EXEC #6:c=0,e=238,p=0,cr=2,cu=30,mis=0,r=10,dep=1,og=4,tim=1293353992515931
EXEC #6:c=0,e=252,p=0,cr=2,cu=32,mis=0,r=10,dep=1,og=4,tim=1293353992516229
EXEC #6:c=0,e=231,p=0,cr=2,cu=30,mis=0,r=10,dep=1,og=4,tim=1293353992516507
EXEC #6:c=0,e=227,p=0,cr=2,cu=30,mis=0,r=10,dep=1,og=4,tim=1293353992516782
EXEC #6:c=0,e=244,p=0,cr=2,cu=32,mis=0,r=10,dep=1,og=4,tim=1293353992517072
EXEC #6:c=0,e=219,p=0,cr=2,cu=30,mis=0,r=10,dep=1,og=4,tim=1293353992517337
EXEC #6:c=0,e=236,p=0,cr=3,cu=30,mis=0,r=10,dep=1,og=4,tim=1293353992517622
EXEC #6:c=0,e=235,p=0,cr=2,cu=30,mis=0,r=10,dep=1,og=4,tim=1293353992517921
EXEC #6:c=0,e=229,p=0,cr=2,cu=30,mis=0,r=10,dep=1,og=4,tim=1293353992518196
EXEC #6:c=0,e=246,p=0,cr=2,cu=32,mis=0,r=10,dep=1,og=4,tim=1293353992518487
EXEC #6:c=0,e=234,p=0,cr=2,cu=30,mis=0,r=10,dep=1,og=4,tim=1293353992518767
EXEC #6:c=6999,e=570,p=0,cr=2,cu=30,mis=0,r=10,dep=1,og=4,tim=1293353992519383

那是 Oracle 针对 CHILD 为它在 PARENT 中删除的每条记录发出删除语句。

一个不同的问题是两者中哪一个更有效:

DELETE FROM CHILD WHERE PARENT_ID = 1;
DELETE FROM PARENT WHERE PARENT_ID = 1;

DELETE FROM PARENT WHERE PARENT_ID = 1;

两者都启用了on delete cascade。令人惊讶的是,在上述第一种情况下,Oracle 将探测子表上的外键索引,以查看是否存在需要级联的行。如果不存在行,Oracle 不会执行级联删除。

【讨论】:

那么答案在哪里?什么更有效。 @magulla:与 Oracle 中的大多数东西一样,这取决于。在删除之前手动删除许多子行可能更有效,但没有什么能阻止另一个会话在此过程中插入更多子行。考虑到如果存在子表,Oracle 会为删除的每个父行发出删除操作,如果性能至关重要,那么您最好用一条语句删除子表感兴趣的行将是一个经验法则,但不能保证“更好”的性能.【参考方案2】:

你不能像这样比较这两个选项。这不是性能问题,而是更多的设计和结构问题。

如果您使用主键/外键设计数据库,则使用级联删除比手动搜索在哪个列和表上有外键并生成匹配的 SQL 语句更容易。

cascading-deletes 功能的主要优点是它允许您减少执行删除操作所需的 SQL 语句数量

【讨论】:

【参考方案3】:

如果你想级联删除并且没有定义外键,你可以使用这样的东西:

DELETE FROM my_table
 WHERE ROWID IN
     ( SELECT ROWID
         FROM my_table
        START WITH (condition_on_the_row_that_you_want_to_delete)
      CONNECT BY PRIOR (primary_key) = (self_foreign_key)
     )

【讨论】:

这种解决方案的缺点是,运行删除的会话没有看到的任何插入的行仍将被插入 - 并且是孤立的,因为您已经杀死了父行,但子行仍然被插入。 @AdamMusch 你提到的场景有解决方案吗?

以上是关于Oracle级联删除的主要内容,如果未能解决你的问题,请参考以下文章

数据库中啥是“级联更新关联字段”和“级联删除关联字段”

java:Oracle(级联删除,左右内外交叉自然连接,子查询,all,any,in)

sql级联更新和级联删除不起作用

如何理解access设置中的“级联更新”和“级联删除”?

sql 级联删除问题

oracle级联查询的问题