今天在改造以前旧项目时出现了一项BUG,是由于以前不规范的EF写法所导致。异常信息如下:

"An entity object cannot be referenced by multiple instances of IEntityChangeTracker(一个实体对象不能由多个 IEntityChangeTracker 实例引用)"

这个问题其实很容易定位,是因为在程序中

使用了不同的DbContext来追踪同一个实体

以下的Demo代码可以轻松地引发该异常:

            using (var dbContext = new ADbContext())
{var aa = dbContext.ClassA.Where(p => p.Id == ).FirstOrDefault();
dbContext.Entry<ClassA>(aa).State = System.Data.Entity.EntityState.Modified;
using (var db2 = new ADbContext())
{
db2.Entry<ClassA>(aa).State = System.Data.Entity.EntityState.Modified;
dbContext.SaveChanges();
}
dbContext.SaveChanges();
}

注意ClassA一定要具有  导航属性  ,如下:

    public class ClassA

    {
public int Id { get; set; }
public string guid { get; set; }
public int? child_id { get; set; } [ForeignKey("child_id")]
public virtual ClassB child { get; set; }
}

多数场景下,一个设计好的系统中都会使用Uow(工作单元)来保证每一次请求使用同一个DbContext(也会有一些系统会需要MSDTC,不在讨论范围内),任意新建DbContext不但容易引发异常,还有可能因为释放不及时而导致内存问题。

要解决上面的这个异常很简单——使用同一个DbContext就行了。

可问题是:

为什么这个项目在以前的运行过程中没有引发错误?

探究


还是上面那个项目,我们把导航属性的 Virtual 去掉,如下:

    public class ClassA

    {
public int Id { get; set; }
public string guid { get; set; }
public int? child_id { get; set; } [ForeignKey("child_id")]
public ClassB child { get; set; }
}

运行项目,你会发现异常不见了。

究竟是什么原因?

对EF稍微了解的同学都知道,EF的导航属性默认是开启延迟加载的。

不了解的同学请搜索关键字补充一下关于EF延迟加载的基础知识,英语能力不错的同学请浏览官方文档( 关联与导航属性加载关联实体

去掉了virtual后,关闭了该导航属性的延迟加载功能,然后异常消失了。

是因为延迟加载导致的这个异常吗?

延迟加载又和IEntityChangeTracker有什么关系呢?

原因


顾名思义,其实 IEntityChangeTracker 就是用来追踪实体信息的,但令人不解的是,为什么关闭延迟加载之后,就算实体同时被两个DbContext追踪也不会报错。

来考虑一下EF的延迟加载是如何实现的,

EF使用了与Castle类似的动态代理技术,同时也存在着相同的缺陷(无法拦截没有被标识为virtual的成员)。

由于没看过EF的源码,官方文档也没有详细的说明,所以我只能推测,IEntityChangeTracker其实发挥类似拦截器的功能,

调用DbContext时,由EF产生实体的动态代理,在访问导航属性时,拦截请求访问数据库并填充导航属性。

而动态代理在产生之后,就无法在Attach到其他的DbContext中。

基于这个推测,我们可以使用一下的代码进行测试,关闭DbContext动态代理。

            using (var dbContext = new ADbContext())
{
dbContext.Configuration.ProxyCreationEnabled = false;
var aa = dbContext.ClassA.Where(p => p.Id == ).FirstOrDefault();
dbContext.Entry<ClassA>(aa).State = System.Data.Entity.EntityState.Modified;
using (var db2 = new ADbContext())
{
db2.Entry<ClassA>(aa).State = System.Data.Entity.EntityState.Modified;
dbContext.SaveChanges();
}
dbContext.SaveChanges();
}

结果通过。

为了证明该异常其实跟延迟加载没有关系,我们可以开启动态代理,然后关闭延迟加载。

            using (var dbContext = new ADbContext())
{
dbContext.Configuration.ProxyCreationEnabled = true;
dbContext.Configuration.LazyLoadingEnabled = false;
var aa = dbContext.ClassA.Where(p => p.Id == ).FirstOrDefault();
dbContext.Entry<ClassA>(aa).State = System.Data.Entity.EntityState.Modified;
using (var db2 = new ADbContext())
{
db2.Entry<ClassA>(aa).State = System.Data.Entity.EntityState.Modified;
dbContext.SaveChanges();
}
dbContext.SaveChanges();
}

异常依旧发生。

证明引发该异常的并不是延迟加载功能,而在于EF动态代理的对象只能由一个DbContext追踪。

还有一点值得一提的是:

如果一个实体没有导航属性的话,EF也不会生成它的动态代理。

EF异常探究(An entity object cannot be referenced by multiple instances of IEntityChangeTracker.)的更多相关文章

  1. EntityFramework 异常 -- An entity object cannot be referenced by multiple instances of IEntityChangeTracker

    问题      在调用 DbSet 的 Attach()  方法时(与将 Entity 设置为 EntityState.Unchanged 状态等价)报告以下错误:      An entity ob ...

  2. An entity object cannot be referenced by multiple instances of IEntityChangeTracker 的解决方案

    使用EF对建立了关系的表新增记录时出现: An entity object cannot be referenced by multiple instances of IEntityChangeTra ...

  3. An entity object cannot be referenced by multiple instances of IEntityChangeTracker.

    如果你和我一样遇到了这个问题,那么你就要检查你要操作的Model对象查询,更新操作的数据库上下文也就是DBContext是否一致.如果不一致也就是说你用AContext去查如AContext.SET& ...

  4. EBS OAF开发中的Java 实体对象(Entity Object)验证功能补充

    EBS OAF开发中的Java 实体对象(Entity Object)验证功能补充 (版权声明,本人原创或者翻译的文章如需转载,如转载用于个人学习,请注明出处:否则请与本人联系,违者必究) EO理论上 ...

  5. 记录一个EF连接查询的异常:the entity or complex type 'x' cannot be constructed in a linq to entities query

    问题解决连接:https://stackoverflow.com/questions/5325797/the-entity-cannot-be-constructed-in-a-linq-to-ent ...

  6. EF中System.Data.Entity.Internal.AppConfig的类型初始值设定项引发异常

    使用Entity的时候遇到的一个错 问题出在项目的App.config中 解决: 1.configSections要写在最顶端 2. 当中的incariantName会变成incariantNodeN ...

  7. APS.NET MVC + EF (02)---ADO.NET Entity FrameWork

    2.1 Entity Framework简介 Ado.net Entity Framework 是Microsoft推出的ORM框架. 2.1.1 什么是ORM 对象关系映射(Object Relat ...

  8. 怎么搭建EF的环境?(Entity Framework)

    1.EF是什么? EF是.net封装的一个用于数据库交互的实体层框架,它的全称是Entity Framework. 2.EF搭建: 新建之后,我们就可以看到里面的内容: 我们可以分别看一下它里面有些什 ...

  9. ASP.NET Core ActionFilter引发的一个EF异常

    最近在使用ASP.NET Core的时候出现了一个奇怪的问题.在一个Controller上使用了一个ActionFilter之后经常出现EF报错. InvalidOperationException: ...

随机推荐

  1. 201521123045 《JAVA程序设计》 第14周学习总结

    1. 本周学习总结 1.1 以你喜欢的方式(思维导图或其他)归纳总结多数据库相关内容. 2. 书面作业 1. MySQL数据库基本操作 建立数据库,将自己的姓名.学号作为一条记录插入.(截图,需出现自 ...

  2. 多线程面试题系列(2): CreateThread与_beginthreadex本质区别

    本文将带领你与多线程作第一次亲密接触,并深入分析CreateThread与_beginthreadex的本质区别,相信阅读本文后你能轻松的使用多线程并能流畅准确的回答CreateThread与_beg ...

  3. [js高手之路]Node.js+jade+mongodb+mongoose实现爬虫分离入库与生成静态文件

    接着这篇文章[js高手之路]Node.js+jade抓取博客所有文章生成静态html文件继续,在这篇文章中实现了采集与静态文件的生成,在实际的采集项目中, 应该是先入库再选择性的生成静态文件.那么我选 ...

  4. Servlet第七篇【Cookie和Session的区别、应用】

    Session和Cookie的区别 从存储方式上比较 Cookie只能存储字符串,如果要存储非ASCII字符串还要对其编码. Session可以存储任何类型的数据,可以把Session看成是一个容器 ...

  5. Spring @RequestParam乱码问题

    在网上找了很多资料才找到解决的方法,通过URL传递命名参数,后台接收的却是乱码解决方法如下: 方法一:将接收的参数重新编码 @RequestMapping(value="/handle&qu ...

  6. 排查Linux机器是否已经被入侵

    随着开源产品的越来越盛行,作为一个Linux运维工程师,能够清晰地鉴别异常机器是否已经被入侵了显得至关重要,个人结合自己的工作经历,整理了几种常见的机器被黑情况供参考 背景信息:以下情况是在CentO ...

  7. 自己把jar包添加到maven仓库中

    定制库到Maven本地资源库 这里有2个案例,需要手动发出Maven命令包括一个 jar 到 Maven 的本地资源库. 要使用的 jar 不存在于 Maven 的中心储存库中. 您创建了一个自定义的 ...

  8. popOver 弹出框简单使用

    1.仿QQ弹出框 1.1用到的知识点 1.1.1如何调整弹出框的大小(这里弹出的也是控制器) 这里已经有讲解过http://blog.csdn.net/iostiannan/article/detai ...

  9. 移植u-boot-2012.04.01到JZ2440

    开发环境:Ubuntu 12.04 开发板:JZ2440  256M NandFlash  64M SDRAM 交叉编译器:arm-linux-gcc-4.3.2 u-boot:u-boot-2012 ...

  10. Django内置的用户认证

    认证登陆 在进行用户登陆验证的时候,如果是自己写代码,就必须要先查询数据库,看用户输入的用户名是否存在于数据库中: 如果用户存在于数据库中,然后再验证用户输入的密码,这样一来就要自己编写大量的代码. ...