.Net中的内存泄露

说明:

虽然已经有GC垃圾回收器在工作,但是还是会出现内存泄露。

内存碎片

费托管内存泄露比托管内存泄露更加严重。GC可以移动托管内存,为其他对象腾空间。但是非托管内存将永远的卡在它的位置。

GC的工作原理

GC遍历所有GC Root对象,将其标记为不回收;然后GC转到他们引用的所有对象,并将它们标记为不回收。最后GC回收剩下的所有内容。

GC Root

GCRoot包括:正在运行的线程的实时堆栈;静态变量;通过interop传递到COM对象的托管对象。

起因:

  1. 对象资源被引用,但是却未被使用!就因为它们被引用了,所以GC不会清理它们,这样它们将永久占用内存。

例如:注册了一个事件+=;但是不注销-=,就会发生这种情况。这被专业的称为:托管内存泄露。

  1. 当通过某种方式分配非托管内存,就没有GC参与了,因此也不会进行释放。.Net中本身就有很多类会分配非托管内存。几乎所有涉及到流、图形、文件系统或网络调用的操作都会在背后分配非托管内存。这些类通常会提供Dispose()方法来进行内存释放。
  2. .Net提供如Marshal/PInvoke这些特殊的类来实现非托管内存的分配。

八种内存泄露

八种内存泄露情况,前6种是托管内存泄露,后2种是非托管内存泄露。

订阅Events

.Net中的Events因导致内存泄露而出名。起因:订阅事件后,该事件对象将保留对你相应类的引用。除非使用不捕获类成员的匿名方法。

假设wifiManager的寿命超过MyClass,那么就已经造成了内存泄露。wifiManager会引用MyClass的任何实例,且GC不会回收它们。

解决方法:

  1. 注销该事件的订阅-=;
  2. 使用弱句柄weak-handler模式。
  3. 使用匿名函数进行订阅,且不要捕获任何类成员。

在匿名方法中捕获类成员

虽然事件机制需要引用一个对象,但是引用对象在匿名方法中捕获类成员时却不明显。

上面的匿名方法中捕获了_id,因此该实例也会被引用。只意味着这个匿名方法还会引用MyClass实例。

可以通过定义局部变量,将_id赋值给localId,此时该匿名方法将不会捕获MyClass了,就可以避免潜在的内存泄露。

静态变量

静态变量及其引用的所有内容都不会被GC回收。

缓存功能

无限次的缓存,最终将耗尽内存。

Var

解决方法:

  1. 删除一段时间未使用的缓存;
  2. 限制缓存大小;
  3. 使用弱引用来保存缓存对象。

错误的WPF绑定

WPF 中的绑定实际上可能会导致内存泄露。通常绑定到DP依赖属性上和实现了INotifyPropertyChanged属性上时是安全的。否则WPF会创建从静态变量到绑定源VM VIEWMODEL的强引用,从而导致内存泄露。

也就是说,定义属性时,要实现INotifyPropertyChanged接口,这会通知WPF不要创建强引用。

另一个是绑定到集合Collection时,也会发生内存泄露。所以要绑定的集合必须要实现INotifyCollectionChanged接口。可以使用现成实现了该接口的observableCollection来避免此问题。

永不终止的线程

GC是不会回收实时堆栈的,实时堆栈中包括正在运行的线程中所有的局部变量和调用堆栈的成员。

出于某种因素,需要创建一个永远运行的线程如Timer,这可能会造成内存泄露。

如果你并没有真正的停止这个timer,那么他会在一个独立的线程中运行,并且timer中用到的各种对象实例是不会被GC回收的。

没有回收非托管内存

非托管内存需要手动编码来回收,而不仅仅是避免不必要的引用。

上述方法使用Marshal.AllocHGlobal方法分配了非托管内存缓冲区。

AllocHGlobal会调用Kernel32.dll中的LocalAlloc方法。如果没有手动调用Marshal.FreeHGlobal来回收内存,则该缓冲区内存将被视为占用了进程的内存堆,从而导致内存泄露。

添加了Dispose方法却不调用它

为了避免这种情况发生,可以使用using语句

Using(var instance=new MyClass){…//do what you want}

编译器会将上面的代码转换为:

使用Dispose Pattern

这种模式可确保即使没有调用Dispose,Dispose也将在实例被垃圾回收时被调用。另一方面,如果调用了Dispose,则finalizer将被抑制(SuppressFinalize)。抑制finalizer很重要,因为finalizer开销很大并且会导致性能问题。

然而,dispose-pattern不是万无一失的。如果从未调用Dispose并且由于托管内存泄漏而导致你的类没有被垃圾回收,那么非托管资源也将不会被释放。

.Net中的内存泄露的更多相关文章

  1. 查找并修复Android中的内存泄露—OutOfMemoryError

    [编者按]本文作者为来自南非约翰内斯堡的女程序员 Rebecca Franks,Rebecca 热衷于安卓开发,拥有4年安卓应用开发经验.有点完美主义者,喜爱美食. 本文系国内ITOM管理平台 One ...

  2. 利用Instrument Leak来发现App中的内存泄露

    XCode提供了一组用于检测内存,调试动画,布局等的工具.对于调试一些性能问题,内存问题非常方便.这里我们使用Leak来发现代码中的内存泄露. 在Leak中启动我们的应用开始监控: 注意,在监控的时候 ...

  3. Android内存优化8 内存检测工具2 LeakCanary——直白的展现Android中的内存泄露

    之前碰到的OOM问题,终于很直白的呈现在我的眼前:我尝试了MAT,但是发现不怎么会用.直到今天终于发现了这个新工具: 当我们的App中存在内存泄露时会在通知栏弹出通知: 当点击该通知时,会跳转到具体的 ...

  4. LeakCanary——直白的展现Android中的内存泄露

    之前碰到的OOM问题,终于很直白的呈现在我的眼前:我尝试了MAT,但是发现不怎么会用.直到今天终于发现了这个新工具: 当我们的App中存在内存泄露时会在通知栏弹出通知: 当点击该通知时,会跳转到具体的 ...

  5. Java中的内存泄露 和 JVM GC(垃圾回收机制)

    一.什么是Java中的内存泄露? 在Java中,内存泄漏就是存在一些被分配的对象,这些对象有下面两个特点, 首先,这些对象是可达的,即在有向图中,存在通路可以与其相连:其次,这些对象是无用的,即程序以 ...

  6. Qt应用中检测内存泄露——VLD

    本文简要描述一下在Qt应用中使用VLD来检测内存泄露.本次测试环境:QtCreator2.3 + Qt4.7.4-vs2008 + VS2008 Express. 1.下载并安装:VLD-2.2: h ...

  7. C++中避免内存泄露常见的解决方式

    常见内存泄露及解决方式-选自ood启发录 new/delete, array new/arrray delete匹配 case 1: 在类的构造函数与析构函数中没有匹配地调用 new/delete! ...

  8. [lua] mac上如何编译snapshot(检测Lua中的内存泄露)

    最近我们的unity手游频繁闪退,只要进入战斗场景,之后一段时间就会闪退,如果是在unity编辑器中则会报出not enough memory的错误!猜测应该是有内存泄漏: 由于我们使用了tolua, ...

  9. Java中的内存泄露的几种可能

    Java内存泄漏引起的原因: 内存泄漏是指无用对象(不再使用的对象)持续占有内存或无用对象的内存得不到及时释放,从而造成内存空间的浪费称为内存泄漏. 长生命周期的对象持有短生命周期对象的引用就很可能发 ...

  10. VS2008中捕获内存泄露(转)

    内存泄露十分讨厌,捕获内存泄露更加令人厌烦…… 其实,VS本身就有内存泄露的检测机制.只需做以下操作即可开启.(同时必须在debug模式 下运行程序并且以 正常流程退出 ) // 在入口函数cpp中添 ...

随机推荐

  1. Flutter 屏幕采集如何实现(提供示例demo)

    在视频会议.线上课堂.游戏直播等场景,屏幕共享是一个最常见的功能.屏幕共享就是对屏幕画面的实时共享,端到端主要有几个步骤:录屏采集.视频编码及封装.实时传输.视频解封装及解码.视频渲染.一般来说,实时 ...

  2. Flask结合gunicorn和nginx反向代理的生产环境部署及踩坑记录

    前言 之前自己写的flask使用gunicorn上线生产环境没有什么问题,但是最近搭建了一个现成的flask项目,当使用python直接运行时不会有问题,而使用gunicorn时则会出现一些问题. 部 ...

  3. LeanCloud 国内域名解析问题,博客评论及阅读数显示失败

    近日,LeanCloud 国内域名解析存在问题,个人博客基于LeanCloud构建的评论及阅读数显示失败. 个人博客地址 关于 LeanCloud 国内域名解析问题的情况更新(6 月 21 日) 声明 ...

  4. 2023年icpc大学生程序设计竞赛-crf

    第一次在除郑轻以外的校外的地方比赛,也是第一次出市比赛,赛程也比较长.20号出发的时候遇到一些意外,不过无伤大雅,第一天热身赛平平无奇,晚上的时候补了一下前年icpc的题,一个多小时做了五题,很是自信 ...

  5. shell 默认参数

    #!/bin/bash dst_dir=${2:-/tmp} # 当 $2 为空或null时,设置默认值. docker cp prometheus:$1 $dst_dir

  6. IIC、SPI、UART三者对比

    下面将对比三者的各自差异: 参考资料: 1.(112条消息) UART, SPI, IIC的详解及三者的区别和联系_iic spi uart_静思心远的博客-CSDN博客

  7. 数据库是要拿来用的,不是用来PK先进性的

    周五参加了WAIC后又和一家上海本地的数据库厂商交流了一下午.等我要买高铁票回南京的时候已经买不到票了.好不容易刷到一张到苏州北的高铁票,我就上了车.上车后突然想起还不如就回苏州老家住一晚算了.到家后 ...

  8. windows传输文件到linux

    PFSTP 打开该软件,在安装putty自带的 连接服务器 open 192.168.142.131 按提示输入账户密码 传送文件 put C:\Users\13662\nifi-1.13.2-bin ...

  9. APubPlat 一款Devops自动化部署、持续集成、堡垒机开源项目、友好的Web Terminal

    嗨.很高心你能进入这里,我是zane,  在这里给你介绍一款完整的Devops自动化部署工具 APubPlat - 一款完整的Devops自动化部署.持续集成.堡垒机.并且友好的Web Termina ...

  10. python教程 入门学习笔记 第1天 初识python python语言环境安装 python编写器

    初识python 一.python语言简介: 1.起源:1989年由荷兰的前谷歌程序员吉多.范罗苏姆(龟叔)创造,python的命名来源于英国电视喜剧Monty Python's Flying Cir ...