.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. spring cloud zuul实践

    一. 描述 Spring Cloud Zuul是基于Netflix开源的Zuul项目构建而成,它作为微服务架构中的网关服务,主要用于实现动态路由.负载均衡和请求过滤等功能. 动态路由:Zuul根据预设 ...

  2. 快速打开指定目录的cmd

    待解决问题 在使用dirsearch.sqlmap等工具时需要进入对应的目录再打开cmd,现要实现配合utools快速打开对应目录的cmd 解决办法:创捷快捷方式 在对应的文件夹下创建一个快捷方式,对 ...

  3. IoTOS-v1.5.3 新增 智能诊断&会话记录导出

    IoTOS v1.5.3 一.新增智能诊断 智能诊断功能: 智能诊断会根据不同上游接口能力开放提供接近官方甚至比官方更加完善的智能诊断功能. 目前还原OneLink官方智能诊断功能包括动效.诊断建议等 ...

  4. Python 运行 shell 命令的一些方法

    哈喽大家好,我是咸鱼 我们知道,python 在自动化领域中被广泛应用,可以很好地自动化处理一些任务 就比如编写 Python 脚本自动化执行重复性的任务,如文件处理.数据处理.系统管理等需要运行其他 ...

  5. LNMP环境搭建wordPress

    LNMP linux Nginx --apache Mysql php 在redhat上LNMP的搭建 首先配置软件源仓库 给redhat选择ISO镜像并连接 输入下面代码后:wq保存并退出 [Bas ...

  6. 2023河南省ICPC大学生程序设计竞赛-wh

    第一次出去比赛,首先感谢程老师选择我们新生更多的比赛机会,感谢! 在周六我们一起做了高铁出发取洛阳参加icpc河南省赛,不得不说洛阳师范学院确实环境很好看..在热身赛时,已经被泼了冷水,这C也太难了, ...

  7. 2023-07-19:布尔表达式 是计算结果不是 true 就是 false 的表达式 有效的表达式需遵循以下约定: ‘t‘,运算结果为 true ‘f‘,运算结果为 false ‘!(subExpr

    2023-07-19:布尔表达式 是计算结果不是 true 就是 false 的表达式 有效的表达式需遵循以下约定: 't',运算结果为 true 'f',运算结果为 false '!(subExpr ...

  8. Windows查询进程和杀死进程

    查询进程 查询进程占用的端口(通过端口查询) netstat -ano |findstr "端口号" 杀死进程 通过进程号查看所属进程名称 tasklist |findstr &q ...

  9. 【Spring】@RequestBody的实现原理

    @RequestBody注解可以用于POST请求接收请求体中的参数,使用方式如下: @Controller public class IndexController { @PostMapping(va ...

  10. debian 防火墙命令 nft

    参考链接:nftables # which nft /usr/sbin/nft # dpkg -S /usr/sbin/nft nftables: /usr/sbin/nft # dpkg -L nf ...