嘉兴中小企业网站建设,cms wordpress 企业,合肥推广优化公司,金融贷款和网站建设哪个赚钱写了一个window服务#xff0c;循环更新sqlite记录#xff0c;内存一点点稳步增长。三天后#xff0c;内存溢出。于是#xff0c;我从自己的代码入手#xff0c;查找到底哪儿占用内存释放不掉#xff0c;最终明确是调用servicestack.ormlite更新sqlite数据库造成的。至于… 写了一个window服务循环更新sqlite记录内存一点点稳步增长。三天后内存溢出。于是我从自己的代码入手查找到底哪儿占用内存释放不掉最终明确是调用servicestack.ormlite更新sqlite数据库造成的。至于是不是框架问题可能性不大因为本地模拟执行的代码没有任何问题。我觉得应该是orm在执行数据库更新后对象还在被引用造成的。这里我贴出一个伪代码我的猜测到底对不对呢现在还不知道。不过在探寻答案的时候对GC的相关机制详细地了解了一遍。一、什么是GC官网中有这么一句话The garbage collector is a common language runtime component that controls the allocation and release of managed memory。原来GC是CLR的一个组件它控制内存的分配与释放。二、托管堆和CLR堆管理器我们知道c#中的引用类型分配在堆上。所谓的堆就是一大块连续的内存地址。CLR堆管理器负责内存的分配、释放。堆又分为小对象堆和大对象堆。它的内存分配流程如下 图片来源《.NET高级调试》pdfCLR加载时就会分配堆。 三、GC的工作机制 GC有三个假设1、如果没有特别声明所有的对象都是垃圾通过引用追踪对象是否为垃圾2、假设托管堆上所有的对象的活跃时间都是短暂的相对于长久活跃的对象来说GC将更频繁地收集短暂活跃的对象3、通过代跟踪对象的持续时间以下是官方文档给出的和这三个假设一致The garbage collector in the common language runtime supports object aging using generationsObjects created more recently are part of newer generations, and have lower generation numbers than objects created earlier in the application life cycle. Objects in the most recent generation are in generation 0. This implementation of the garbage collector supports three generations of objects, generations 0, 1, and 2每代都有自己的堆假如0代的堆满了就会触发GC然后把依然有引用的对象升级放到1代对象。最后压缩堆把剩余的堆空间合并到一块。1代对象也是如此操作。但到了2代就处理不同了。2代的堆可能是大对象堆它的压缩代价过于高昂所以只是合并相邻的空间。 图片来源博客园c#技术漫谈之垃圾回收GCGarbage collection happens automatically when a request for memory cannot be satisfied using available free memory GC发生的时机就是相应的堆达到了阈值因为堆也有大小限制并不是无限的。尽管2代堆或者大对象堆满的时候通过增加新的内存段来满足内存分配如果没有可用的内存这时就会报内存溢出。四、GC不能释放非托管资源有两种情况第一种托管代码引用了非托管资源比如文件操作、数据库连接、网络连接等。这时候必须手动释放或实现 dispose模式或实现对象终结When a type uses unmanaged resources that must be released before instances of the type are reclaimed, the type can implement a finalizer.In most cases, finalizers are implemented by overriding the Object.Finalize method; however, types written in C# or C implement destructors, which compilers turn into an override of Object.Finalize 必须注意的一点是实现对象终结器GC会在释放对象之前自动调用。其实这是一个代价非常高昂的备用机制。所以能自己释放非托管资源的就自己释放。如果一个对象中包含有终结器那么在new的时候放入到终结者队列。当GC会把这个对象标为垃圾时放入到另一个队列F-Reachable中。这个队列包含了所有带有终结器并且将被作为垃圾收集的对象这些对象的终结器都将被执行。在垃圾收集的过程总并不会执行终结器代码。而是由.NET 进程的终结线程调用。因此此时的垃圾回收滞后一段时间目的在于等待终结器代码执行的完成。五、dispose模式这是基类和子类的dispose模式来源于官网。原文地址:https://www.cnblogs.com/wangqiang3311/p/10280000.html.NET社区新闻深度好文欢迎访问公众号文章汇总 http://www.csharpkit.com