假设在数据库中有两个表:Person表和Book表,Person和Book是一对多关系

Person表数据:

Book表数据:

可以看到数据库Book表中所有的数据都属于Person表中"F"这个人

Person表,下面的Person类是该表在EF Core中的实体类型:

public partial class Person
{
public Person()
{
Book = new HashSet<Book>();
} public int Id { get; set; }
public string Code { get; set; }
public string Name { get; set; }
public DateTime? CreateTime { get; set; }
public DateTime? UpdateTime { get; set; } public ICollection<Book> Book { get; set; }
}

因为一个Person对应多个Book,所以Person类中有个集合属性public ICollection<Book> Book { get; set; }

此外Person类的Code属性为EF Core中实体的Key属性:

modelBuilder.Entity<Person>(entity =>
{
entity.HasKey(e => e.Code);//设置Code为EF Core实体Person的Key属性 entity.HasIndex(e => e.Code)
.HasName("IX_Person")
.IsUnique(); entity.Property(e => e.Id).HasColumnName("ID"); entity.Property(e => e.Code)
.IsRequired()
.HasMaxLength(); entity.Property(e => e.CreateTime)
.HasColumnType("datetime")
.HasDefaultValueSql("(getdate())"); entity.Property(e => e.Name).HasMaxLength(); entity.Property(e => e.UpdateTime).HasColumnType("datetime");
});

Book表,下面的Book类是该表在EF Core中的实体类型:

public partial class Book
{
public int Id { get; set; }
public string BookCode { get; set; }
public string BookName { get; set; }
public string PersonCode { get; set; } public Person PersonCodeNavigation { get; set; }
}

同样因为一个Book只对应一个Person,所以Book类中有个导航属性public Person PersonCodeNavigation { get; set; }

此外Book类的BookCode属性为EF Core中实体的Key属性:

modelBuilder.Entity<Book>(entity =>
{
entity.HasKey(e => e.BookCode);//设置BookCode为EF Core实体Book的Key属性 entity.Property(e => e.Id).HasColumnName("ID");
entity.Property(e => e.Id).ValueGeneratedOnAddOrUpdate(); entity.Property(e => e.BookCode).HasMaxLength(); entity.Property(e => e.BookName).HasMaxLength(); entity.Property(e => e.PersonCode).HasMaxLength(); entity.HasOne(d => d.PersonCodeNavigation)
.WithMany(p => p.Book)
.HasPrincipalKey(p => p.Code)
.HasForeignKey(d => d.PersonCode)
.OnDelete(DeleteBehavior.Cascade)
.HasConstraintName("FK_Book_Person");
});

现在假如我们想通过new一个Book b3,然后在Person类的集合属性Book中来删除b3,结果数据库Book表中的数据并不会被删除,原因见下面注释:

using (TestDBContext testDBContext = new TestDBContext())
{
var person = testDBContext.Person.Where(P => P.Code == "F").Include(p => p.Book).First(); Book b3 = new Book() { BookCode = "B3" }; var f = person.Book.Remove(b3);//不起作用,结果BookCode为"B3"的数据并没有从数据库Book表中删除,这是因为我们在上面的testDBContext.Person.Where(P => P.Code == "F").Include(p => p.Book).First()中调用了Include方法,导致BookCode为"B3"的实体已经被加载到EF Core的DbContext中被Track了,这时上面一行的Book b3 = new Book() { BookCode = "B3" }其BookCode也为"B3",Book b3将无法在person.Book.Remove(b3)时被Attach到DbContext内进行Track,因为DbContext中被Track的实体必须是Key属性值唯一的,前面已经有一个BookCode为"B3"实体实例在DbContext中了,就无法再加入另一个BookCode为"B3"的实体实例了,所以删除不会起任何作用。
//此外这里的删除不起作用还有个很重要的原因,那就是person.Book.Remove(b3)方法会先尝试在位于内存中的person.Book集合中匹配Book b3,但是由于Book b3是我们自己new出来的,它和person.Book中的集合成员指向的是不同的Book对象实例,所以Book b3无法和person.Book中的任何集合成员匹配,尽管它们的Key属性BookCode都为"B3"。由于Book b3和person.Book的集合成员匹配都失败了,那么person.Book.Remove(b3)不会删除数据库中的任何数据,只有当以Book b3和person.Book的一个集合成员匹配成功时,person.Book.Remove(b3)才会在数据库中删除对应的数据。 testDBContext.SaveChanges();
}

接下来,我们通过从Person.Book集合中找出BookCode为"B3"的实体实例赋值给Book b3,然后从Person.Book集合中再删除该实例,这次我们会发现数据库Book表中的数据就被删除了,原因见下面注释:

using (TestDBContext testDBContext = new TestDBContext())
{
var person = testDBContext.Person.Where(P => P.Code == "F").Include(p => p.Book).First(); Book b3 = person.Book.First(b => b.BookCode == "B3"); EntityState s = testDBContext.Entry(b3).State;//状态为EntityState.Unchanged var f = person.Book.Remove(b3);//起作用,因为这时上面一行的Book b3指向的就是testDBContext.Person.Where(P => P.Code == "F").Include(p => p.Book).First()中Include方法加载的实体实例,所以相当于Book b3已经被加载到EF Core的DbContext中被Track了,所以这里调用person.Book.Remove(b3)之后Book b3的State将被标记为EntityState.Deleted,然后执行下面的testDBContext.SaveChanges()后,从数据库Book表中将BookCode为 "B3"的数据行删除掉 s = testDBContext.Entry(b3).State;//状态为EntityState.Deleted testDBContext.SaveChanges();
}

注意采用Person.Book集合删除Book b3这种方式,会根据EF Core中DeleteBehavior的不同设置会有不同的结果,因为本例中我们设置了DeleteBehavior.Cascade,所以EF Core会从数据库Book表中将BookCode为 "B3"的数据行删除掉,但是如果采用DeleteBehavior的其它设置,那么会得到不同的结果,详情可以查看:Cascade Delete

最后我们来看看new一个Book b3,通过DbContext的DbSet属性来删除b3,同样这次数据库Book表中的数据也会被成功删除,原因见下面注释:

using (TestDBContext testDBContext = new TestDBContext())
{
Book b3 = new Book() { BookCode = "B3" }; EntityState s = testDBContext.Entry(b3).State;//状态为EntityState.Detached
var f = testDBContext.Book.Remove(b3);//起作用,因为testDBContext.Book.Remove(b3)是通过DbContext的DbSet<Book>属性来删除数据的,在此之前也没有BookCode为"B3"的实体实例被加载到DbContext进行Track,所以这里调用testDBContext.Book.Remove(b3)会把Book b3加载到DbContext进行Track,并将Book b3的State标记为EntityState.Deleted,然后执行下面的testDBContext.SaveChanges()后,从数据库Book表中将BookCode为 "B3"的数据行删除掉 s = testDBContext.Entry(b3).State;//状态为EntityState.Deleted testDBContext.SaveChanges();
}

EF Core中如何通过实体集合属性删除从表的数据的更多相关文章

  1. EF Core中,通过实体类向SQL Server数据库表中插入数据后,实体对象是如何得到数据库表中的默认值的

    我们使用EF Core的实体类向SQL Server数据库表中插入数据后,如果数据库表中有自增列或默认值列,那么EF Core的实体对象也会返回插入到数据库表中的默认值. 下面我们通过例子来展示,EF ...

  2. EF Core中怎么实现自动更新实体的属性值到数据库

    我们在开发系统的时候,经常会遇到这种需求数据库表中的行被更新时需要自动更新某些列. 数据库 比如下面的Person表有一列UpdateTime,这列数据要求在行被更新后自动更新为系统的当前时间. Pe ...

  3. [小技巧]EF Core中如何获取上下文中操作过的实体

    原文地址:https://www.cnblogs.com/lwqlun/p/10576443.html 作者:Lamond Lu 源代码:https://github.com/lamondlu/EFC ...

  4. EF Core 中多次从数据库查询实体数据,DbContext跟踪实体的情况

    使用EF Core时,如果多次从数据库中查询一个表的同一行数据,DbContext中跟踪(track)的实体到底有几个呢?我们下面就分情况讨论下. 数据库 首先我们的数据库中有一个Person表,其建 ...

  5. EF Core 中DbContext不会跟踪聚合方法和Join方法返回的结果,及FromSql方法使用讲解

    EF Core中: 如果调用Queryable.Count等聚合方法,不会导致DbContext跟踪(track)任何实体. 此外调用Queryable.Join方法返回的匿名类型也不会被DbCont ...

  6. EF Core中如何正确地设置两张表之间的关联关系

    数据库 假设现在我们在SQL Server数据库中有下面两张表: Person表,代表的是一个人: CREATE TABLE [dbo].[Person]( ,) NOT NULL, ) NULL, ...

  7. EF Core 2.0 已经支持自动生成父子关系表的实体

    现在我们在SQL Server数据库中有Person表如下: CREATE TABLE [dbo].[Person]( ,) NOT NULL, ) NULL, ) NULL, ) NULL, [Cr ...

  8. EF Core 三 、 骚操作 (导航属性,内存查询...)

    EF Core 高阶操作 本文之前,大家已经阅读了前面的系列文档,对其有了大概的了解 我们来看下EF Core中的一些常见高阶操作,来丰富我们业务实现,从而拥有更多的实现选择 1.EF 内存查找 wh ...

  9. EF Core中避免贫血模型的三种行之有效的方法(翻译)

    Paul Hiles: 3 ways to avoid an anemic domain model in EF Core 1.引言 在使用ORM中(比如Entity Framework)贫血领域模型 ...

随机推荐

  1. Java温故而知新(5)设计模式详解(23种)

    一.设计模式的理解 刚开始“不懂”为什么要把很简单的东西搞得那么复杂.后来随着软件开发经验的增加才开始明白我所看到的“复杂”恰恰就是设计模式的精髓所在,我所理解的“简单”就是一把钥匙开一把锁的模式,目 ...

  2. 合并excel的多个sheet

    '合并excel的多个sheetSub 合并当前工作簿下的所有工作表()Application.ScreenUpdating = FalseFor j = 1 To Sheets.Count If S ...

  3. 织梦后台添加友情链接的方法(flink标签)

    标记名称:flink[标签简介][功能说明]:用于获取友情链接,其对应后台文件为"includetaglibflink.lib.php".[适用范围]:全局标记,适用V55,V56 ...

  4. Raspberry 安装vstudio

    Visual Studio Code微软公司推出的一款轻量级的Visual Studio风格的跨平台的IDE.当然,除了Windows,OSX,还能在树莓派上使用.目前树莓派上可用的IDE真不多,VS ...

  5. 红黑树(R-B Tree)

    R-B Tree简介 R-B Tree,全称是Red-Black Tree,又称为“红黑树”,它一种特殊的二叉查找树.红黑树的每个节点上都有存储位表示节点的颜色,可以是红(Red)或黑(Black). ...

  6. tcp.h

    /* * Copyright (c) 1991-1997 Regents of the University of California. * All rights reserved. * * Red ...

  7. windows下安装jmeter

    windows下安装jmeter post by rocdk890 / 2012-8-19 16:08 Sunday windows技术 发表评论 JMeter是Apache软件基金会的产品,用于对静 ...

  8. 【SQL重温】面试之数据库基础练习

    简介 最近在练习SQL基础,首先感叹一下,在机器上写和在纸上写还是有区别的. 本文的练习题目请点击此链接进行查看:http://www.cnblogs.com/edisonchou/p/3878135 ...

  9. SQL 查询:查询学生平均成绩

    编程萌新,因为遇到这么个SQL 查询的问题:在一张表A里有如下字段:学生姓名.学科名.学科成绩.写一条SQL 语句查出各科平均成绩并按学生姓名分组,按如下格式显示:学生姓名|语文|数学|英语.一开始遇 ...

  10. Python初学者第八天 元组和字典

    8day 1.数据类型:元组 元组:有序的,不可变地数据的集合.但若包含其他可变元素,这些元素可变.显示的告诉别人,此处不可修改: a = (1,2,3,4,5,['1','a']) 2.数据类型:字 ...