73

我想禁用具有实体框架代码优先的链接表的级联删除。例如,如果许多用户有很多角色,而我尝试删除一个角色,我希望阻止该删除,除非当前没有与该角色关联的用户。我已经删除了我的级联删除约定OnModelCreating

protected override void OnModelCreating(DbModelBuilder modelBuilder) {
    ...
    modelBuilder.Conventions.Remove<OneToManyCascadeDeleteConvention>();

然后我设置了用户角色链接表:

modelBuilder.Entity<User>()
    .HasMany(usr => usr.Roles)
    .WithMany(role => role.Users)
    .Map(m => {
        m.ToTable("UsersRoles");
        m.MapLeftKey("UserId");
        m.MapRightKey("RoleId");
    });

然而,当 EF 创建数据库时,它会为外键关系创建删除级联,例如。

ALTER TABLE [dbo].[UsersRoles]  WITH CHECK ADD  CONSTRAINT [FK_dbo.UsersRoles_dbo.User_UserId] FOREIGN KEY([UserId])
REFERENCES [dbo].[User] ([UserId])
ON DELETE CASCADE
GO

ALTER TABLE [dbo].[UsersRoles]  WITH CHECK ADD  CONSTRAINT [FK_dbo.UsersRoles_dbo.Role_RoleId] FOREIGN KEY([RoleId])
REFERENCES [dbo].[Role] ([RoleId])
ON DELETE CASCADE
GO

如何停止 EF 生成此删除级联?

4

3 回答 3

116

我得到了答案。:-) 那些级联删除是因为ManyToManyCascadeDeleteConvention. 您需要删除此约定以防止它为链接表创建级联删除:

modelBuilder.Conventions.Remove<ManyToManyCascadeDeleteConvention>();
于 2012-12-04T14:53:44.150 回答
6

我认为在全球范围内关闭ManyToManyCascadeDeleteConvention并不是一个明智的选择。相反,最好只为相关表关闭它。

这可以通过为 property 编辑生成的迁移文件来实现cascadeDelete。例如:

AddForeignKey("dbo.UsersRoles", "UserId", "dbo.User", "UserId", cascadeDelete: false);

于 2017-12-06T13:23:32.500 回答
5

我同意 Ebram Khalil 的观点,即为一张桌子关闭它是一个不错的选择。但是,我喜欢尽可能接近自动构建的迁移,所以我会在 OnModelCreating 中设置它:

modelBuilder.Entity<User>()
    .HasMany(usr => usr.Roles)
    .WithMany(role => role.Users)
    .Map(m => {
        m.ToTable("UsersRoles");
        m.MapLeftKey("UserId");
        m.MapRightKey("RoleId");
    })
    .WillCascadeOnDelete(false);

我相信这会保留另一个方向的删除,因此如果两者都需要被阻止(在本例中有意义),则需要从以下开始进行类似的调用Entity<User>(Role)

当然,这是在提出问题之后的很长时间。因此,它可能在 2012 年无效。

于 2018-01-09T22:14:31.420 回答