级联删除
Entity Framework Core (EF Core) 表示使用外键的关系。 具有外键的实体是关系中的子实体或依赖实体。 此实体的外键值必须与相关主体/父实体的主键值(或替换键值)匹配。
如果删除主体/父实体,则依赖项/子项的外键值将不再匹配任何主体/父实体的主键或替换键。 这是无效状态,将导致在大多数数据库中出现引用约束冲突。
可通过两种方法来避免此引用约束冲突:
- 将外键值设置为 null
- 同时删除依赖实体/子实体
第一个选项仅适用于其中外键属性(及其映射到的数据库列)必须可为 null 的可选关系。
第二个选项适用于任何类型的关系,它被称作“级联删除”。
提示
本文档从更新数据库的角度介绍级联删除(和删除孤立项)。 本文大量使用在在 EF Core 中更改跟踪和更改外键和导航文章中介绍的概念。 请确保在此处处理材料之前充分了解这些概念。
提示
通过从 GitHub 下载示例代码,你可运行并调试到本文档中的所有代码。
发生级联行为时
当依赖实体/子实体无法再与其当前主体/父实体关联时,需要执行级联删除。 发生这种情况的原因可能是主体/父实体已被删除,或者当主体/父实体仍存在,但依赖实体/子实体不再与其关联时。
删除主体/父实体
请考虑此简单模型,其中 Blog
是与 Post
(依赖实体/子实体)的关系中的主体/父实体。 Post.BlogId
是一个外键属性,其值必须与该文章所属博客中的 Blog.Id
主键匹配。
public class Blog
{
public int Id { get; set; }
public string Name { get; set; }
public IList<Post> Posts { get; } = new List<Post>();
}
public class Post
{
public int Id { get; set; }
public string Title { get; set; }
public string Content { get; set; }
public int BlogId { get; set; }
public Blog Blog { get; set; }
}
按照约定,由于 Post.BlogId
外键属性是不可为 null 的,因此该关系被配置为必需的。 默认情况下,所需的关系配置为使用级联删除。 要详细了解建模关系,请参阅关系。
删除博客时,所有文章都将被级联删除。 例如:
using var context = new BlogsContext();
var blog = await context.Blogs.OrderBy(e => e.Name).Include(e => e.Posts).FirstAsync();
context.Remove(blog);
await context.SaveChangesAsync();
SaveChanges 以 SQL Server 为例,生成以下 SQL:
-- Executed DbCommand (1ms) [Parameters=[@p0='1'], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
DELETE FROM [Posts]
WHERE [Id] = @p0;
SELECT @@ROWCOUNT;
-- Executed DbCommand (0ms) [Parameters=[@p0='2'], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
DELETE FROM [Posts]
WHERE [Id] = @p0;
SELECT @@ROWCOUNT;
-- Executed DbCommand (2ms) [Parameters=[@p1='1'], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
DELETE FROM [Blogs]
WHERE [Id] = @p1;
SELECT @@ROWCOUNT;
断开关系
我们不会删除博客,而是断开每篇文章与其博客之间的关系。 为此,可将每篇文章的引用导航 Post.Blog
设置为 null:
using var context = new BlogsContext();
var blog = await context.Blogs.OrderBy(e => e.Name).Include(e => e.Posts).FirstAsync();
foreach (var post in blog.Posts)
{
post.Blog = null;
}
await context.SaveChangesAsync();
还可通过从 Blog.Posts
集合导航中删除每篇文章内容来断开关系:
using var context = new BlogsContext();
var blog = await context.Blogs.OrderBy(e => e.Name).Include(e => e.Posts).FirstAsync();
blog.Posts.Clear();
await context.SaveChangesAsync();
无论哪种情况,结果都一样:没有删除博客,但是删除了不再与任何博客关联的文章:
-- Executed DbCommand (1ms) [Parameters=[@p0='1'], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
DELETE FROM [Posts]
WHERE [Id] = @p0;
SELECT @@ROWCOUNT;
-- Executed DbCommand (0ms) [Parameters=[@p0='2'], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
DELETE FROM [Posts]
WHERE [Id] = @p0;
SELECT @@ROWCOUNT;
删除不再与任何主体/依赖实体关联的实体这一行为被称作“删除孤立项”。
提示
级联删除和删除孤立项是密切相关的。 当断开与所需的主体/父实体之间的关系时,两者都将导致删除依赖实体/子实体。 对于级联删除,由于主体/父实体本身已删除,因此发生了这种断开。 对于孤立项,主体/父实体仍然存在,但不再与依赖实体/子实体相关。
发生级联行为的位置
可将级联行为应用于:
- 当前 DbContext 跟踪的实体
- 数据库中尚未加载到上下文中的实体
级联删除被跟踪实体
EF Core 始终将配置的级联行为应用于跟踪的实体。 这意味着如上面的示例所示,如果应用程序将所有相关的依赖实体/子实体加载到 DbContext 中,则无论如何配置数据库,都将正确应用级联行为。
提示
可使用 ChangeTracker.CascadeDeleteTiming 和 ChangeTracker.DeleteOrphansTiming 控制在被跟踪实体上发生级联行为的确切时间。 有关详细信息,请参阅更改外键和导航。
数据库中的级联删除
许多数据库系统还提供在数据库中删除实体时触发的级联行为。 使用 EnsureCreated 或 EF Core 迁移创建数据库时,EF Core 会根据 EF Core 模型中的级联删除行为来配置这些行为。 例如,通过上述模型,使用 SQL Server 时将为文章创建下表:
CREATE TABLE [Posts] (
[Id] int NOT NULL IDENTITY,
[Title] nvarchar(max) NULL,
[Content] nvarchar(max) NULL,
[BlogId] int NOT NULL,
CONSTRAINT [PK_Posts] PRIMARY KEY ([Id]),
CONSTRAINT [FK_Posts_Blogs_BlogId] FOREIGN KEY ([BlogId]) REFERENCES [Blogs] ([Id]) ON DELETE CASCADE
);
请注意,定义博客和文章之间关系的外键约束是用 ON DELETE CASCADE
配置的。
如果我们知道数据库是这样配置的,那么我们可以删除博客,而无需先加载文章,数据库将负责删除与此博客相关的所有文章。 例如:
using var context = new BlogsContext();
var blog = await context.Blogs.OrderBy(e => e.Name).FirstAsync();
context.Remove(blog);
await context.SaveChangesAsync();
请注意,文章没有 Include
,因此它们不会被加载。 在这种情况下,SaveChanges 将仅删除博客,因为这是唯一正在跟踪的实体:
-- Executed DbCommand (6ms) [Parameters=[@p0='1'], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
DELETE FROM [Blogs]
WHERE [Id] = @p0;
SELECT @@ROWCOUNT;
如果未针对级联删除配置数据库中的外键约束,则将导致异常。 但在这种情况下,数据库删除了文章,因为它在创建时是用 ON DELETE CASCADE
配置的。
注意
数据库通常没有任何自动删除孤立项的方法。 这是因为虽然 EF Core 使用导航以及外键来表示关系,但是数据库仅具有外键而没有导航。 这意味着通常无法在不将双方都加载到 DbContext 的情况下断开关系。
注意
EF Core 内存中数据库当前不支持数据库中的级联删除。
警告
软删除实体时,请勿在数据库中配置级联删除。 这可能会导致实体被意外删除,而不是软删除。
数据库级联限制
一些数据库(最突出的是 SQL Server)对形成周期的级联行为有限制。 例如,请考虑以下模型:
public class Blog
{
public int Id { get; set; }
public string Name { get; set; }
public IList<Post> Posts { get; } = new List<Post>();
public int OwnerId { get; set; }
public Person Owner { get; set; }
}
public class Post
{
public int Id { get; set; }
public string Title { get; set; }
public string Content { get; set; }
public int BlogId { get; set; }
public Blog Blog { get; set; }
public int AuthorId { get; set; }
public Person Author { get; set; }
}
public class Person
{
public int Id { get; set; }
public string Name { get; set; }
public IList<Post> Posts { get; } = new List<Post>();
public Blog OwnedBlog { get; set; }
}
该模型具有 3 个关系,所有这些关系都是必需的,因此按约定配置为级联删除:
- 删除博客将级联删除所有相关文章
- 删除文章的作者将导致作者的文章被级联删除
- 删除博客所有者将导致该博客被级联删除
这一切都是合理的(不过在博客管理策略中有些苛刻!),但是尝试创建配置了这些级联的 SQL Server 数据库会导致以下异常:
Microsoft.Data.SqlClient.SqlException (0x80131904):将 FOREIGN KEY 约束“FK_Posts_Person_AuthorId”引入表“Posts”可能会导致循环或多重级联路径。 请指定 ON DELETE NO ACTION 或 ON UPDATE NO ACTION,或修改其他 FOREIGN KEY 约束。
有两种方法可处理这种情况:
- 将一个或多个关系更改为不级联删除。
- 配置数据库,但不包含这些级联删除中的一个或多个,然后确保已加载所有依赖实体,以便 EF Core 可执行级联行为。
在我们的示例中采用第一种方法,我们可通过为文章与博客之间的关系赋予可为 null 的外键属性来使其成为可选关系:
public int? BlogId { get; set; }
可选关系使得即使没有博客,文章也可存在,这意味着默认情况下将不再配置级联删除。 这表示级联操作不再循环,并且可以在 SQL Server 上创建数据库而不会出现错误。
采取第二种方法,我们可以保持必需的博客所有者关系并对其配置来进行级联删除,但是使此配置仅适用于跟踪的实体,而不适用于数据库:
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder
.Entity<Blog>()
.HasOne(e => e.Owner)
.WithOne(e => e.OwnedBlog)
.OnDelete(DeleteBehavior.ClientCascade);
}
现在,如果我们同时加载某用户及其拥有的博客,然后删除该用户,会发生什么呢?
using var context = new BlogsContext();
var owner = await context.People.SingleAsync(e => e.Name == "ajcvickers");
var blog = await context.Blogs.SingleAsync(e => e.Owner == owner);
context.Remove(owner);
await context.SaveChangesAsync();
EF Core 将级联删除所有者,以便博客也被删除:
-- Executed DbCommand (8ms) [Parameters=[@p0='1'], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
DELETE FROM [Blogs]
WHERE [Id] = @p0;
SELECT @@ROWCOUNT;
-- Executed DbCommand (2ms) [Parameters=[@p1='1'], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
DELETE FROM [People]
WHERE [Id] = @p1;
SELECT @@ROWCOUNT;
但是,如果在删除所有者时未加载博客:
using var context = new BlogsContext();
var owner = await context.People.SingleAsync(e => e.Name == "ajcvickers");
context.Remove(owner);
await context.SaveChangesAsync();
则由于违反数据库中的外键约束,将引发异常:
Microsoft.Data.SqlClient.SqlException:DELETE 语句与 REFERENCE 约束“FK_Blogs_People_OwnerId”冲突。 数据库 "Scratch"、表 "dbo.Blogs"、列 "OwnerId" 中发生冲突。 语句已终止。
级联 NULL
可选关系将可为 null 的外键属性映射到可为 null 的数据库列。 这意味着当删除当前主体/父实体或断开与依赖实体/子实体的关系时,可将外键值设置为 NULL。
让我们再看一下发生级联行为时的示例,但这次可选关系由可为 null 的 Post.BlogId
外键属性表示:
public int? BlogId { get; set; }
删除每篇文章的相关博客时,该文章的外键属性将设置为 NULL。 例如,此代码与之前的代码相同:
using var context = new BlogsContext();
var blog = await context.Blogs.OrderBy(e => e.Name).Include(e => e.Posts).FirstAsync();
context.Remove(blog);
await context.SaveChangesAsync();
现将在调用 SaveChanges 时导致以下数据库更新:
-- Executed DbCommand (2ms) [Parameters=[@p1='1', @p0=NULL (DbType = Int32)], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
UPDATE [Posts] SET [BlogId] = @p0
WHERE [Id] = @p1;
SELECT @@ROWCOUNT;
-- Executed DbCommand (0ms) [Parameters=[@p1='2', @p0=NULL (DbType = Int32)], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
UPDATE [Posts] SET [BlogId] = @p0
WHERE [Id] = @p1;
SELECT @@ROWCOUNT;
-- Executed DbCommand (1ms) [Parameters=[@p2='1'], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
DELETE FROM [Blogs]
WHERE [Id] = @p2;
SELECT @@ROWCOUNT;
同样,如果使用上述任一示例来断开关系:
using var context = new BlogsContext();
var blog = await context.Blogs.OrderBy(e => e.Name).Include(e => e.Posts).FirstAsync();
foreach (var post in blog.Posts)
{
post.Blog = null;
}
await context.SaveChangesAsync();
或:
using var context = new BlogsContext();
var blog = await context.Blogs.OrderBy(e => e.Name).Include(e => e.Posts).FirstAsync();
blog.Posts.Clear();
await context.SaveChangesAsync();
则在调用 SaveChanges 时,将使用 NULL 外键值更新文章:
-- Executed DbCommand (2ms) [Parameters=[@p1='1', @p0=NULL (DbType = Int32)], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
UPDATE [Posts] SET [BlogId] = @p0
WHERE [Id] = @p1;
SELECT @@ROWCOUNT;
-- Executed DbCommand (0ms) [Parameters=[@p1='2', @p0=NULL (DbType = Int32)], CommandType='Text', CommandTimeout='30']
SET NOCOUNT ON;
UPDATE [Posts] SET [BlogId] = @p0
WHERE [Id] = @p1;
SELECT @@ROWCOUNT;
请参阅更改外键和导航,详细了解 EF Core 如何在外键和导航的值更改时管理外键和导航。
注意
自 2008 年首版以来,实体框架默认情况下都会修复这类关系。 在 EF Core 之前,它没有名称,且无法更改。 它现在称为 ClientSetNull
,如下一部分所述。
当删除可选关系中的主体/父实体时,数据库也可配置为级联 NULL。 但是,与在数据库中使用级联删除相比,这种情况要少得多。 在使用 SQL Server 时,在数据库中同时使用级联删除和级联 NULL 几乎总是会导致关系循环。 若要详细了解如何配置级联 NULL,请参阅下一部分。
配置级联行为
提示
请务必阅读上述部分,然后再转到此处。 如果不了解上述资料,那么配置选项可能没有意义。
使用 OnModelCreating 中的 OnDelete 方法按关系配置级联行为。 例如:
protected override void OnModelCreating(ModelBuilder modelBuilder)
{
modelBuilder
.Entity<Blog>()
.HasOne(e => e.Owner)
.WithOne(e => e.OwnedBlog)
.OnDelete(DeleteBehavior.ClientCascade);
}
若要详细了解如何配置实体类型之间的关系,请参阅关系。
OnDelete
从公认地令人混淆的 DeleteBehavior 枚举中接受一个值。 该枚举既定义了 EF Core 在跟踪实体上的行为,又定义了使用 EF 创建架构时数据库中级联删除的配置。
对数据库架构的影响
下表显示了由 EF Core 迁移或 EnsureCreated 创建的外键约束上每个 OnDelete
值的结果。
DeleteBehavior | 对数据库架构的影响 |
---|---|
Cascade | ON DELETE CASCADE |
限制 | ON DELETE RESTRICT |
NoAction | 数据库默认值 |
SetNull | ON DELETE SET NULL |
ClientSetNull | 数据库默认值 |
ClientCascade | 数据库默认值 |
ClientNoAction | 数据库默认值 |
关系数据库中 ON DELETE NO ACTION
(数据库默认值)和 ON DELETE RESTRICT
的行为通常相同或非常相似。 尽管 NO ACTION
可能意味着什么,但这两个选项都会导致强制执行引用约束。 区别是当有一个时,是影响数据库何时检查约束。 请查看数据库文档,了解数据库系统上 ON DELETE NO ACTION
和 ON DELETE RESTRICT
之间的具体区别。
SQL Server 不支持 ON DELETE RESTRICT
,因此改为使用 ON DELETE NO ACTION
。
导致数据库级联行为的唯一值是 Cascade
和 SetNull
。 所有其他值会将数据库配置为不级联任何更改。
对 SaveChanges 行为的影响
以下各部分中的表格介绍了删除主体/父实体或断开与主体/子实体的关系时,依赖实体/子实体所发生的情况。 每张表都涵盖下述内容之一:
- 可选(可为 null 的外键)和必需(不可为 null 的外键)关系
- 依赖项/子项何时由 DbContext 加载和跟踪,以及它们何时仅存在于数据库中
与已加载的依赖项/子项的必需关系
DeleteBehavior | 删除主体/父实体时 | 断开与主体/父实体的关系时 |
---|---|---|
Cascade | EF Core 删除的依赖项 | EF Core 删除的依赖项 |
限制 | InvalidOperationException |
InvalidOperationException |
NoAction | InvalidOperationException |
InvalidOperationException |
SetNull | SqlException (创建数据库时) |
SqlException (创建数据库时) |
ClientSetNull | InvalidOperationException |
InvalidOperationException |
ClientCascade | EF Core 删除的依赖项 | EF Core 删除的依赖项 |
ClientNoAction | DbUpdateException |
InvalidOperationException |
注意:
- 这种必需关系的默认值为
Cascade
。 - 调用 SaveChanges 时,对必需关系使用除级联删除以外的其他方法将导致异常。
- 通常,这是来自 EF Core 的
InvalidOperationException
,因为在已加载的子项/依赖项中检测到无效状态。 ClientNoAction
会强制 EF Core 在将依赖项发送到数据库之前不检查修复它们,因此在这种情况下,数据库将引发异常,然后由 SaveChanges 将其包装在DbUpdateException
中。- 创建数据库时会拒绝
SetNull
,因为外键列不可为 null。
- 通常,这是来自 EF Core 的
- 由于已加载依赖项/子项,因此它们始终会被 EF Core 删除,并且永远不会留下来等到数据库被删除。
与未加载的依赖项/子项的必需关系
DeleteBehavior | 删除主体/父实体时 | 断开与主体/父实体的关系时 |
---|---|---|
Cascade | 数据库删除的依赖项 | 不可用 |
限制 | DbUpdateException |
不可用 |
NoAction | DbUpdateException |
不可用 |
SetNull | SqlException (创建数据库时) |
不可用 |
ClientSetNull | DbUpdateException |
不可用 |
ClientCascade | DbUpdateException |
不可用 |
ClientNoAction | DbUpdateException |
不可用 |
注意:
- 此处没法断开关系,因为未加载依赖项/子项。
- 这种必需关系的默认值为
Cascade
。 - 调用 SaveChanges 时,对必需关系使用除级联删除以外的其他方法将导致异常。
- 通常,这是
DbUpdateException
,理由是未加载依赖项/子项,因此数据库只能检测到无效状态。 然后,SaveChanges 会将数据库异常包装在DbUpdateException
中。 - 创建数据库时会拒绝
SetNull
,因为外键列不可为 null。
- 通常,这是
与已加载的依赖项/子项的可选关系
DeleteBehavior | 删除主体/父实体时 | 断开与主体/父实体的关系时 |
---|---|---|
Cascade | EF Core 删除的依赖项 | EF Core 删除的依赖项 |
限制 | EF Core 将依赖外键设置为 NULL | EF Core 将依赖外键设置为 NULL |
NoAction | EF Core 将依赖外键设置为 NULL | EF Core 将依赖外键设置为 NULL |
SetNull | EF Core 将依赖外键设置为 NULL | EF Core 将依赖外键设置为 NULL |
ClientSetNull | EF Core 将依赖外键设置为 NULL | EF Core 将依赖外键设置为 NULL |
ClientCascade | EF Core 删除的依赖项 | EF Core 删除的依赖项 |
ClientNoAction | DbUpdateException |
EF Core 将依赖外键设置为 NULL |
注意:
- 这种可选关系的默认值为
ClientSetNull
。 - 永远不会删除依赖项/子项,除非配置了
Cascade
或ClientCascade
。 - 所有其他值都会导致 EF Core 将依赖外键设置为 NULL...
- ...
ClientNoAction
除外,它指示 EF Core 在删除主体/父实体时不处理依赖项/子项的外键。 因此,数据库会引发异常,由 SaveChanges 将其包装为DbUpdateException
。
- ...
与未加载的依赖项/子项的可选关系
DeleteBehavior | 删除主体/父实体时 | 断开与主体/父实体的关系时 |
---|---|---|
Cascade | 数据库删除的依赖项 | 不可用 |
限制 | DbUpdateException |
不可用 |
NoAction | DbUpdateException |
不可用 |
SetNull | 数据库将依赖外键设置为 NULL | 不可用 |
ClientSetNull | DbUpdateException |
不可用 |
ClientCascade | DbUpdateException |
不可用 |
ClientNoAction | DbUpdateException |
不可用 |
注意:
- 此处没法断开关系,因为未加载依赖项/子项。
- 这种可选关系的默认值为
ClientSetNull
。 - 除非已将数据库配置为级联删除或级联 NULL,否则必须加载依赖项/子项以避免数据库异常。