4

我有一个巨大的遗留数据库,其中一个表有多个外键指向另一个表,并且看不到一个级联,类似于下面的示例表:

create table Users (
    Id int primary key identity,
    Name varchar(max)
)

create table Products (
    Id int primary key identity,
    Name varchar(max),
    CreatedBy int foreign key references Users(Id),
    UpdatedBy int foreign key references Users(Id)
)

insert into Users values('Bar')
insert into Users values('Baz')
insert into Products values('Foo', 1, 2)

我需要能够删除一些旧数据,但它当然会引发引用异常:

delete from Users where Name='Bar'

DELETE 语句与 REFERENCE 约束“FK__Products__Create__1AD3FDA4”冲突。冲突发生在数据库“Foo”、表“dbo.Products”、列“CreatedBy”中。


由于数据库的复杂性,我无法预先删除所有引用,因此我尝试以编程方式添加临时外键,并设置级联来解决它们。但是,对于这个具有多个指向另一个表的外键的特定表,这会导致cycles or multiple cascade paths第二个UpdatedBy更改:

alter table Products add foreign key (CreatedBy) references Users(Id) on delete cascade
alter table Products add foreign key (UpdatedBy) references Users(Id) on delete cascade

在表“Products”上引入 FOREIGN KEY 约束“FK__Products__Update__1DB06A4F”可能会导致循环或多个级联路径。指定 ON DELETE NO ACTION 或 ON UPDATE NO ACTION,或修改其他 FOREIGN KEY 约束。


我怎样才能delete from Users where在保持参照完整性的同时工作,无论是通过某种方式解决多个级联路径问题还是其他方式?

4

1 回答 1

5

我个人不会这样做(我会预先删除所有引用的数据并手动检查完整性)。请参阅:可以使用 T-SQL 临时禁用外键约束吗?

引用:

-- disable all constraints
EXEC sp_msforeachtable "ALTER TABLE ? NOCHECK CONSTRAINT all"

禁用约束后删除您的数据,但请记住之后再次打开它们!

-- enable all constraints
EXEC sp_msforeachtable @command1="print '?'", @command2="ALTER TABLE ? WITH CHECK CHECK CONSTRAINT all"

另请注意,存储过程sp_msforeachtable未记录在案,可能会在 SQL Server 的未来版本中消失。

如果您不想完全禁用约束(也许您有一个适用的表列表),那么只需禁用它们,如您在上面的代码中所见。

ALTER TABLE [Products] NOCHECK CONSTRAINT ALL
DELETE FROM [Users] where Name='Bar'
ALTER TABLE [Products] WITH CHECK CHECK CONSTRAINT ALL

所有功劳归功于克里斯托夫的回答。请投票!

于 2012-12-19T11:57:50.760 回答