查到一些解决方案:          

 

1, http://www.codethinked.com/keep-your-iqueryable-in-check

自定义常用方法,屏蔽IQuery功能。这个好像有点靠谱。但麻烦。

 

2,http://stackoverflow.com/questions/123057/how-do-i-avoid-a-memory-leak-with-linq-to-sql

这个看起来是应对 ObjectContext的。

 

3,http://stackoverflow.com/questions/19116851/entity-framework-using-repository-pattern-unit-of-work-and-unity

这个看起来简单些 ,用using.

 

4, 直接用using的方式

using(dbcontext)

{

}

 

5, http://blog.robustsoftware.co.uk/2008/11/clearing-cache-of-linq-to-sql.html

这个针对 DataContext的一个扩展方法。

 

 

最后没有更好的解决办法。

 

 

/******************************Edit 2014-1-25**************************************************************/

 

最后在分析了DbContext的源代码以后,发现了其中的奥秘。

首先DbContext本身是对于ObjectContext的一个包装,在使用IQueryable以后会将Enitiy存于 objectStateManager ,在快速的循环中,_unchangedEntityStore不会被释放,一直被缓存,所以呢?你会发现程序的内存在不断的上涨,这不应该算是内存泄漏。 由于 _unchangedEntityStore 中的值变得越来越大,程序在查询前先去 _unchangedEntityStore 中查找,由于Dictionary<EntityKey, EntityEntry> 在很大的情况下,查询性能会急剧下降,所以程序变慢。 直到慢得你无法忍受。

 

下面是基本代码。

 

public class Class1

 {

     public static void Test()

     {

         Domain.Entities.Models.passportContext context =

             new Domain.Entities.Models.passportContext("passportContext");

         context.Configuration.ProxyCreationEnabled = false;

         context.Configuration.LazyLoadingEnabled = false;

 

         var start = int.MaxValue;

 

 

         var om = ((IObjectContextAdapter)context).ObjectContext.ObjectStateManager;

 

         om.ObjectStateManagerChanged += om_ObjectStateManagerChanged;

 

         while (true)

         {

             var list = context.UserSources.OrderByDescending(x => x.UserId).Where(x => x.UserId < start).Take(100).ToList();

 

 

 

 

             ObjectContext oc = ((IObjectContextAdapter)context).ObjectContext;

 

             var m = oc.ObjectStateManager;

             var a = list.First();

 

             var count = GetUnchangedCount(context);

 

             var mth = m.GetType().GetMethod("GetObjectStateEntriesCount", BindingFlags.Instance | BindingFlags.NonPublic);

 

             var unchanged = mth.Invoke(m, new object[] { EntityState.Unchanged });

 

             if (a.UserId % 4 == 0)

             {

                 a.FromSite += 1;

                 context.Entry(a).State = System.Data.EntityState.Modified;

             }

 

 

             count = GetUnchangedCount(context);

 

             ClearUnchangedCache(context);

 

             Console.WriteLine(a.UserId);

 

             //ClearCache(context);

             start = a.UserId;

         }

 

     }

 

     static void om_ObjectStateManagerChanged(object sender, System.ComponentModel.CollectionChangeEventArgs e)

     {

         Console.WriteLine("[]" + e.Action + "" + e.Element.GetType() + "::" + e.Element);

     }

 

 

     public static void ClearCache(DbContext context)

     {

         ObjectContext oc = DbContext2ObjectContext(context);

 

         const BindingFlags flags = BindingFlags.GetField | BindingFlags.NonPublic | BindingFlags.Instance;

 

         var c = oc.GetType().GetField("_cache", flags);

 

         //var value = c.GetValue(oc);

 

         c.SetValue(oc, null);

 

         //value = c.GetValue(oc);

     }

 

 

     static int GetUnchangedCount(DbContext context)

     {

 

         var objectStateManager = DbContext2ObjectStateManager(context);

 

         var mth = objectStateManager.GetType().GetMethod("GetObjectStateEntriesCount", BindingFlags.Instance | BindingFlags.NonPublic);

 

         var unchanged = mth.Invoke(objectStateManager, new object[] { EntityState.Unchanged });

 

         return (int)unchanged;

     }

 

 

     static ObjectContext DbContext2ObjectContext(DbContext context)

     {

         return ((IObjectContextAdapter)context).ObjectContext;

     }

 

     static ObjectStateManager DbContext2ObjectStateManager(DbContext context)

     {

         return DbContext2ObjectContext(context).ObjectStateManager;

     }

 

     static void ClearUnchangedCache(DbContext context)

     {

         var objectStateManager = DbContext2ObjectStateManager(context);

 

         const BindingFlags flags = BindingFlags.GetField | BindingFlags.NonPublic | BindingFlags.Instance;

 

         var c = objectStateManager.GetType().GetField("_unchangedEntityStore", flags);

 

         c.SetValue(objectStateManager, null);

     }

 

 }

 

 

然后发现内存还是疯长。

 

那就再加一个方法。

static void ClearKeylessEntityCache(DbContext context)

       {

           var objectStateManager = DbContext2ObjectStateManager(context);

 

           const BindingFlags flags = BindingFlags.GetField | BindingFlags.NonPublic | BindingFlags.Instance;

 

           var c = objectStateManager.GetType().GetField("_keylessEntityStore", flags);

 

           c.SetValue(objectStateManager, null);

       }

 

 

再看内存,

 

这样世界顿时安静了下来。

 

 

 

 

 

最后说明:

这个方法的确不是一个好的办法,但是由于EF的对象管理机制决定。 人家缓存也有人家缓存的道理。

实际上MS也没有打算让你按这样的方法来用,

用using(dbcontext)

{

…..

}

这样的方式反而是保持了 Kiss。

 

 

对于这样的“改进” 会不会有什么影响呢?

这个就再开个文章再来测试了。

关于 Linq to EF 的内存泄漏问题的更多相关文章

  1. EntityFramework Core依赖注入上下文方式不同造成内存泄漏了解一下?

    前言 这个问题从未遇见过,是一位前辈问我EF Core内存泄漏问题时我才去深入探讨这个问题,刚开始我比较惊讶,居然还有这种问题,然后就有了本文,直接拿前辈的示例代码并稍加修改成就了此文,希望对在自学E ...

  2. Webservice WCF WebApi 前端数据可视化 前端数据可视化 C# asp.net PhoneGap html5 C# Where 网站分布式开发简介 EntityFramework Core依赖注入上下文方式不同造成内存泄漏了解一下? SQL Server之深入理解STUFF 你必须知道的EntityFramework 6.x和EntityFramework Cor

    Webservice WCF WebApi   注明:改编加组合 在.net平台下,有大量的技术让你创建一个HTTP服务,像Web Service,WCF,现在又出了Web API.在.net平台下, ...

  3. 记一次 WinDbg 分析 .NET 某工厂MES系统 内存泄漏分析

    一:背景 1. 讲故事 上个月有位朋友加微信求助,说他的程序跑着跑着就内存爆掉了,寻求如何解决,截图如下: 从聊天内容看,这位朋友压力还是蛮大的,话说这貌似是我分析的第三个 MES 系统了,看样子 . ...

  4. Java进击C#——应用开发之Linq和EF

    本章简言 上一章笔者对于WinForm开发过程用到的几个知识点做了讲解.笔者们可以以此为开端进行学习.而本章我们来讲一个跟ORM思想有关的知识点.在讲之前让我们想一下关于JAVA的hibernate知 ...

  5. TcpListener的异步调用内存泄漏---最近测试结果,没有泄露

    我后来加大了client的连接/断开的次数(500,1000),Server端的连接被释放了. 这说明: 1. 此代码是 可以正常工作的. 2.TcpListener/TcpListener的asyn ...

  6. Erlang 内存泄漏分析

    随着项目越来越依赖Erlang,碰到的问题也随之增加.前段时间线上系统碰到内存高消耗问题,记录一下troubleshooting的分析过程.线上系统用的是Erlang R16B02版本. 问题描述 有 ...

  7. linq和EF查询的用法和区分

    我们做项目时,难免会遇到用的不知道是啥,及把linq和EF搞混了 今天我带领大家梳理下思路: 首先说linq查询,然后介绍EF查询 1.linq查询 当我们使用linq查询时,转到定义会调到Query ...

  8. 【linux】linux下对java程序生成dump文件,并使用IBM Heap Analyzer进行分析,查找定位内存泄漏的问题代码

    1.首先,java程序启动在linux,怎么生成dump文件? 1>第一步,首先你需要得到java程序的PID,最简单的方法使用如下命令 ps -ef|grep java 或者如果是docker ...

  9. 应用开发之Linq和EF

    本章简言 上一章笔者对于WinForm开发过程用到的几个知识点做了讲解.笔者们可以以此为开端进行学习.而本章我们来讲一个跟ORM思想有关的知识点.在讲之前让我们想一下关于JAVA的hibernate知 ...

随机推荐

  1. 【技术调研】最强Node-RED初探总结

    在某个项目中需要调研下node-red的功能,我大概花了三天时间研究了相关的官方文档,写了几个Demo总结了下node-red相关的功能.如需转载,请注明出处 https://www.cnblogs. ...

  2. 一些c++

    1.static 静态局部对象: 一旦被创建,在程序结束前都不会被撤销.当定义静态局部对象的函数结束时,静态局部对象不会撤销. 2.内联函数: 避免函数调用的开销. 在函数返回类型前加上关键字 inl ...

  3. 通过键盘上下键 JS事件,控制候选词的选择项

    效果图 JS代码 //上下键 选择事件 searchBackgroud 为样式,只做标记,无实质样式,因为和其他样式不兼容,只能添加CSS $(document).keydown(function ( ...

  4. mahout in Action2.2-给用户推荐图书(2)-分析对用户推荐书目的结果

    2.2.3 Analyzing the output 在之前的程序运行结果中我们得到的结果输出是: RecommendedItem [item:104, value:4.257081] 程序要求选择一 ...

  5. 最小的VIM操作指南

    最小VIM操作指南 vim的操作命令非常多,为了能开始工作,必须学会一个最小的vim操作集合,这里做个总结. 1.插入.追加.插入新行 i:在当前光标所在字符的前面插入,当前字符及其后面的字符后撤 a ...

  6. 为cscope查找列表增添色彩

    我在使用cscope的时候,偏好于不用quickfix窗口来显示查找列表,而是选择类似ctag的列表.但这会带来一个比较麻烦的问题,就是窗口列表一片白色,看起来非常难受: 特别是当搜索结果特别多的时候 ...

  7. 【bzoj2705】[SDOI2012]Longge的问题

    2705: [SDOI2012]Longge的问题 Time Limit: 3 Sec  Memory Limit: 128 MBSubmit: 2507  Solved: 1531[Submit][ ...

  8. Apache apachectl命令

    一.简介 apachectl命令是Apache的Web服务器前端控制工具,用以启动.关闭和重新启动Web服务器进程. 二.语法 http://www.jinbuguo.com/apache/menu2 ...

  9. 使用原理视角看 Git

    1. Git 的玩法 欢迎来到 Coding 技术小馆,我叫谭贺贺,目前我在 Coding.net 主要负责 WebIDE 与 Codeinsight 的开发.我今天带来的主要内容是 Git 的原理与 ...

  10. 跨库连接报错Server 'myLinkedServer' is not configured for RPC

    Solution: Problem is most likely that RPC is not configured for your linked server. That is not a de ...