.NET 常见内存泄漏

时间:2022-02-27 20:57:04

简介

以前认为,.NET程序内存都是托管的,如果不是调用非托管资源,应该不会存在内存泄漏的问题,但是,最近两天对归档程序内存使用分析,发现,事情不是想象的那么简单。

 

.NET内存泄漏,更准确的说应该是对象超过生命周期而不能被GC回收。本文列举了几种可能导致内存泄漏的情形,并提供示例代码,及解决方案,希望对大家有所帮助。

 

所举的例子都是经过自己极端简化过的,只是为了说明观点,具体项目中遇到的情形会复杂很多,需要具体分析。

事件注册后未解除注册

示例代码

.NET 常见内存泄漏

程序运行时,内存使用情况如图:

.NET 常见内存泄漏

原因分析

示例代码中,表面看起来,一次循环过后,上次new出来的对象a,不会继续被引用到,应该被视为垃圾,下次垃圾回收的时候,会被回收掉。但是,通过实验证明,对象a是不会被回收掉的。

 

原因是因为,对象b注册了click事件而未销毁,导致对象b会一直引用到a,a不会被视为垃圾被GC回收。

解决办法

对象a生命周期结束后,解除事件的注册,只有这样,a所占用的内存才会在下次垃圾回收的时候被回收掉。

.NET 常见内存泄漏

下图为解除事件注册绑定后,内存的使用情况:

.NET 常见内存泄漏

静态引用

示例代码

.NET 常见内存泄漏

如果一直调用CreateObject方法,程序运行时,内存使用情况如图:

.NET 常见内存泄漏

原因分析

静态变量的生命周期是全局的,即程序不退出,所占用的内存一直不会被释放掉。

解决办法

尽量少用静态成员,若一定要使用,注意静态成员引用的对象是否会一直增长。

控件不使用后未销毁

示例代码

.NET 常见内存泄漏

程序运行时,内存使用情况如图:

.NET 常见内存泄漏

原因分析

示例代码中,控件lbl已经被移除了,表面看起来,下次循环时,该控件应该被视为垃圾,在下次垃圾回收时,被回收掉。但实验证明,内存不会被回收掉。原因在于,控件未被真正销毁掉。

解决办法

控件生命周期结束之后,调用Dispose方法,确保控件能被真正销毁掉。

.NET 常见内存泄漏

下图为调用Dispose方法后,内存的使用情况:

.NET 常见内存泄漏

 

调用非托管资源而未释放

示例代码

.NET 常见内存泄漏

原因分析

使用未托管对象之后忘记释放。

解决办法

.NET 常见内存泄漏

对于实现了IDisposable接口的类可以使用using方式使用,或者显示调用Dispose方法。

工具

日志输出

穷人的工具,输出进程内存使用量 :)

.NET 常见内存泄漏

任务管理器

通过观察任务管理器内存使用情况,查看是否存在内存泄露。

.NET 常见内存泄漏

 

.NET Memory Profiler

专业的.NET性能调优工具,缺点,不是免费的 :)

.NET 常见内存泄漏

参考