使用一个SQL脚本删除子行和父行

时间:2012-03-09 20:12:11

标签: sql oracle parent parent-child

我没有删除子行然后编写另一个sql语句来删除父行,而是想使用一个将同时执行这两个操作的语句。仅供参考:我们使用Oracle数据库。

更新:我没有权利在CASCADE上进行删除

7 个答案:

答案 0 :(得分:15)

使用cascading deletes定义外键。然后你只需要删除“父”行。

答案 1 :(得分:4)

delete from 
(
select * from parent join child using (id)
where id = 1
)

警告!仅删除父行和子行所在的位置。不会删除没有孩子的父母

答案 2 :(得分:2)

你只能做得很糟糕 - 即使用触发器。

create table parent
(pid number,
   constraint parent_pk
     primary key (pid)
     using index
);

create table child
  (cid number,
   pid number,
   constraint child_pk
     primary key(cid)
     using index,
   constraint child_fk
     foreign key (pid)
     references parent (pid)
  );

create index child_fk on child (pid);

create trigger fake_delete_cascade
before delete on parent
for each row
begin
  delete from child where pid = :old.pid;
end;
/

insert into parent values (1);
insert into child values (1,1);
commit;
select count(*) from child;
delete from parent where pid = 1;
select count(*) from child;

答案 3 :(得分:1)

如果您在删除父行时总是要删除子项,则可以声明外键约束,以便Oracle自动删除子项

create table parent (
  parentID number primary key,
  parentData varchar2(100)
);

create table child (
  childID number primary key,
  parentID number references parent( parentID ) on delete cascade,
  childData varchar2(100)
);
例如,

将声明父表和子表,并在删除父行时自动删除子行。如果您不希望自动强制执行此类操作,或者您不喜欢在后台“自动”发生事件时添加的复杂性,那么您可能会使用多个DELETE语句。

答案 4 :(得分:0)

另一种(无聊的方式,我们在数据库中有这个,由于未知原因,不使用外键作为约束 - 是的)这样做是在删除之后(或之前)创建一个触发器。 p>

您必须编写另一个删除查询,但只需在触发器中。

但如果你不能把删除级联,我不确定你可以添加触发器......

答案 5 :(得分:0)

万一它对其他人有帮助,我只是在this Stackoverflow question的帮助下编写了一个PLSQL脚本来对表中的所有外键约束进行此操作。希望对您有所帮助。

DECLARE
  CURSOR constraint_cursor IS SELECT *
                              FROM (SELECT a.table_name,
                                           a.constraint_name,
                                           a.column_name,
                                           c_pk.table_name r_table_name,
                                           b.column_name   r_column_name
                                    FROM user_cons_columns a
                                           JOIN user_constraints c ON a.owner = c.owner
                                                                        AND a.constraint_name = c.constraint_name
                                           JOIN user_constraints c_pk ON c.r_owner = c_pk.owner
                                                                           AND
                                                                         c.r_constraint_name = c_pk.constraint_name
                                           JOIN user_cons_columns b ON C_PK.owner = b.owner
                                                                         AND
                                                                       C_PK.CONSTRAINT_NAME = b.constraint_name AND
                                                                       b.POSITION = a.POSITION
                                    WHERE c.constraint_type = 'R'
                                      and c_pk.owner = 'YOUR SCHEMA HERE') tbl;
  sql_statement VARCHAR2(2048) := NULL;
  tab_row       constraint_cursor%rowtype;
BEGIN
  OPEN constraint_cursor;
  FOR i in 1..80 LOOP
    FETCH constraint_cursor into tab_row;
    EXECUTE IMMEDIATE 'ALTER table ' || tab_row.table_name || ' drop constraint ' || tab_row.constraint_name;
    EXECUTE IMMEDIATE 'ALTER table ' || tab_row.table_name || ' add constraint ' || tab_row.constraint_name || ' FOREIGN KEY (' ||
    tab_row.column_name || ') references ' || tab_row.r_table_name || '(' || tab_row.r_column_name || ') ON DELETE CASCADE ';
  end loop;
  close constraint_cursor;
end;

答案 6 :(得分:0)

在相对较大且极其关键的数据库上对该问题进行了一些非常糟糕的体验之后,我决定为此创建一个Silver bullet!因为找不到any!实际上,该线程中的解决方案/答案都不能满足问题的需求。

请参阅我的github上的CASCADELETE回购。