Entity framework 级联删除注意事项
版本:EF6.0.1 RC
一对多场景,在子对象映射中开启级联删除情况下,删除父对象将自动删除其下所有子对象,需要注意一些事项:
需要保证DbContext中已经加载了该父对象的所有子对象。
如果DbContext内未加载子对象将不级联删除子对象(请看示例代码1),
如DbContext只加载部分子对象也只级联删除这些子对象(遗漏删除未加载的子对象),(请看示例代码2)。
因此在查询父对象只应该使用Include("子对象属性名")查询(请看示例代码3)或者在DbContext另外把其下所有子对象查询出来(请看示例代码4),再进行对父对象的删除方可级联删除子对象。
示例代码:
这里只列出关键实体和过程代码,省去DbContext相关配置代码
//父类 public class Tb2Entity
{
public int Id { get; set; }
public string Name { get; set; }
public virtual IList<Item2Entity> Items { get; set; }
}
//子类
public class Item2Entity
{
public int Id { get; set; }
public int? Pid { get; set; }
public string Title { get; set; }
public virtual Tb2Entity Parent { get; set; }
}
示例代码1(不正确)
只删除了父对象遗漏删除子对象了
var i = 15;
using (var db1 = new Db1Context("db conn string"))
{
var tb1 = new Tb2Entity
{
Id = i,
Name = "n1",
Items = new List<Item2Entity> {
new Item2Entity{Id=1, Title="t1"},
new Item2Entity{Id=2, Title="t2"},
new Item2Entity{Id=3, Title="t3"}
}
};
db1.Set<Tb2Entity>().Add(tb1);
db1.SaveChanges();
}
Console.ReadKey();
using (var db1 = new Db1Context(Conns.db1))
{
var tb1 = db1.Set<Tb2Entity>().Find(i);
db1.Set<Tb2Entity>().Remove(tb1);
db1.SaveChanges();
}
未例代码2(不正确)
只删除了父对象和部分子对象(遗漏删除id=3的子对象)
var i = 15;
using (var db1 = new Db1Context(Conns.db1))
{
var tb1 = new Tb2Entity
{
Id = i,
Name = "n1",
Items = new List<Item2Entity> {
new Item2Entity{Id=1, Title="t1"},
new Item2Entity{Id=2, Title="t2"},
new Item2Entity{Id=3, Title="t3"}
}
};
db1.Set<Tb2Entity>().Add(tb1);
db1.SaveChanges();
}
Console.ReadKey();
using (var db1 = new Db1Context(Conns.db1))
{
var tb1 = db1.Set<Tb2Entity>().Find(i);
var item1 = db1.Set<Item2Entity>().Find(1);
var item2 = db1.Set<Item2Entity>().Find(2);
db1.Set<Tb2Entity>().Remove(tb1);
db1.SaveChanges();
}
示例代码3(正确)
父对象连同子对象一起加载,级联删除时得以一起删除
var i = 15;
using (var db1 = new Db1Context(Conns.db1))
{
var tb1 = new Tb2Entity
{
Id = i,
Name = "n1",
Items = new List<Item2Entity> {
new Item2Entity{Id=1, Title="t1"},
new Item2Entity{Id=2, Title="t2"},
new Item2Entity{Id=3, Title="t3"}
}
};
db1.Set<Tb2Entity>().Add(tb1);
db1.SaveChanges();
}
Console.ReadKey();
using (var db1 = new Db1Context(Conns.db1))
{
var tb1 = db1.Set<Tb2Entity>()
.Include(m => m.Items) //同时加载所有子对象
.Where(m => m.Id == i).Take(1).SingleOrDefault();
db1.Set<Tb2Entity>().Remove(tb1);
db1.SaveChanges();
}
示例代码4(正确)
子对象通过延迟方式加载进来,级联删除时得以一起删除
var i = 15;
using (var db1 = new Db1Context(Conns.db1))
{
var tb1 = new Tb2Entity
{
Id = i,
Name = "n1",
Items = new List<Item2Entity> {
new Item2Entity{Id=1, Title="t1"},
new Item2Entity{Id=2, Title="t2"},
new Item2Entity{Id=3, Title="t3"}
}
};
db1.Set<Tb2Entity>().Add(tb1);
db1.SaveChanges();
}
Console.ReadKey();
using (var db1 = new Db1Context(Conns.db1))
{
var tb1 = db1.Set<Tb2Entity>().Find(i);
//延迟方式加载所有子对象
if (tb1.Items != null)
{
}
db1.Set<Tb2Entity>().Remove(tb1);
db1.SaveChanges();
}
Entity framework 级联删除注意事项的更多相关文章
- Entity Framework 级联删除
为一对主从表增加级联删除功能 protected override void OnModelCreating(DbModelBuilder modelBuilder) { modelBuilder.E ...
- SQL Server 与 Entity Framework 级联删除
SQL Server 级联设置我就不多说了,网上很多教程. 我想提的是 cycles or multiple cascade paths 的问题. 简单的说如果你的级联设置不是一个树型,而是一个带有循 ...
- EF级联删除
引言 在主表中指定Key,子表中指定Required后,并不会在数据库中生成级联删除的外键.那怎么才能使EF在数据中生成级联删除的外键? SQLServer数据库中级联删除功能配置界面: 上图 ...
- 【转】Entity Framework教程
转自:http://www.cnblogs.com/xray2005/category/189491.html Entity Framework系列文章导航 摘要: 本节集合了Entity Fra ...
- Entity Framework 简单增删改操作
前言 在 Entity Framework 简单查询操作 中主要是学习了在Entity Framework中的几种不同模式的查询操作,现在主要来学习一下简单的增加.删除.修改操作. 增加 在EF中添加 ...
- MVC5 Entity Framework学习之实现主要的CRUD功能
在上一篇文章中,我们使用Entity Framework 和SQL Server LocalDB创建了一个MVC应用程序,并使用它来存储和显示数据.在这篇文章中,你将对由 MVC框架自己主动创建的CR ...
- Entity Framework Code First级联删除
如果我们要到一对主从表增加级联删除,则要在主表中的引用属性上增加Required关键字,如: public class Destination { public int DestinationId { ...
- Entity Framework Code First级联删除(转)
使用Data Annotations: 如果我们要到一对主从表增加级联删除,则要在主表中的引用属性上增加Required关键字,如: public class Destination { public ...
- Entity Framework Core系列之DbContext(删除)
上一篇我们介绍了Entity Framework Core系列之DbContext(修改),这一篇我们介绍下删除数据 修改实体的方法取决于context是否正在跟踪需要删除的实体. 下面的示例中con ...
随机推荐
- Linux IO函数的使用和区别
Linux系统中的IO函数主要有read.write.recv.send.recvmsg.sendmsg.readv.writev,本篇主要介绍他们的使用以及区别. read函数: #include ...
- HTML中tr标签设置边框不显示的解决办法
今天在操作表格的时候发现设置表格中行的边框没有显示,然后自己新建了一个表格发现确实不显示 <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Tr ...
- javascript --- Function模式
回调函数 在javascript中,当一个函数A作为另外一个函数B的其中一个参数时,则称A函数为回调函数,即A可以在函数B的运行周期内执行(开始,中间,结束). 举例来说,有一个函数用于生成node. ...
- MSCRM 报表显示 rsprocessingaborted 错误
今天又有朋友遇到rsprocessingaborted这个问题,想想这个已经是很老很老的问题了,得在写一遍补充下. 一.首先会考虑是不是SrsDataConnector没有安装的原因,如果正常安装的话 ...
- Urban Planning and Public Health - Reflection on Professor Webster's article in Urban Planning Forum
1. General review. Professor Webster published this article in Urban Planning Forum, one of the top ...
- Xcode中的常用快捷键
新建项目 com + shift +N 新建文件 com + N 偏好设置 通用 com + , 跳到指定行 com + L 当前行加断点 com + \ 移动编辑区最上方 ...
- Android实现登录
登录界面布局文件 1 <RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android ...
- android加固系列—3.加固前先学会破解,静态修改so
[版权所有,转载请注明出处.出处:http://www.cnblogs.com/joey-hua/p/5138585.html] 项目jni的关键代码(项目地址见文章底部),获取当前程序的包名com. ...
- AndRoid studio创建APP图标
打开---File----New----Image asset 注意:在design页面可能没有image asset选项!必须在其他编辑页面! 这就打开了图标设置页面,找到自己想要的图标就好!下面框 ...
- eclipse编码格式设置
大家好,我是小Alan,很高兴大家能够看到这篇小小的技术点文章,这还是从参加工作以来,小Alan写的第一篇博文.喜欢能够给一些朋友带来方便. 说到eclipse编码格式的设置其实一个非常非常小的事情, ...