.NET资源泄露与处理方案

.NET虽然拥有强大易用的垃圾回收机制,但并非由于这样,你就能够对资源管理听任无论,其实在稍不注意的时候,可能就形成了资源泄露,甚至所以致使系统崩溃,到那时再来排查问题就已是困难重重。html

1、知识点简单介绍数组

常见的资源泄露有:多线程

  • 内存泄漏:非托管资源没有释放、非静态对象注册了静态实例。
  • GDI泄露:字体。
  • 句柄泄露:Socket或线程。
  • 用户对象泄露:移除的对象未释放。

2、具体实例函数

1. 内存泄漏学习

很常见的现象是分不清哪些对象须要释放,对于控件、Stream等一些非托管资源也只管新增,却没有释放,功能是实现了,却埋了颗不小的雷。字体

private void button1_Click(object sender, EventArgs e)
{
    for(int i=0;i<1000;i++)
        this.Controls.Add(new TabPage());
}
private void button1_Click(object sender, EventArgs e)
{
    new Form2.ShowDialog();
}

若是你以为写这样的代码很Cool,很简洁,你在项目中也有这么写代码,那你就碰到大麻烦了,你试试在上面Form2中开个大一点的数组来检查内存,而后运行,按几下按钮,你就会发现,内存一直增长,即便你调用了GC也无济于事。因此,对于此类非托管资源要记住释放,用完即废能够采用using关键字。this

public Form2()
{
    InitializeComponent();
    MyApp.FormChanged += FormChanged;
}

上面这个例子中,MyApp是一个静态类,若是在实例对象中向这种类里面注册了事件,而又没有取消注册,这样也会遇到大麻烦,即便在外部已经记得调用了Form2的Dispose也是没用的。spa

解决方案.net

2. GDI泄露线程

通常会跟字体相关,例如我曾在Android上用Cocos2d作一个小游戏时频繁地切换字体、Dev控件的Font属性赋值也会有这种现象。

XXX.Font = new Font(...)

解决方案

  • 这个问题我目前是采用字体池来解决,相似线程池的概念,相同Key值取同一个对象。如有更好方案欢迎留言讨论

3. 句柄泄露

通常跟Socket和Thread(线程)有关

for(int i=0;i<1000;i++){
    new Thread(()=>{
        Thread.Sleep(1000);
    }).Start();
}

解决方案

  • Socket的场景暂时没遇到。
  • 线程问题采用线程池相关的辅助类能有效解决,例如ThreadPool、Task、Parallel。

4. 用户对象泄露

通常跟移除的对象未释放有关

private void button1_Click(object sender, EventArgs e)
{
    tab.Remove(tabPage);
}

3、最后特别奉送一个内存释放的大招

[DllImport("kernel32.dll", EntryPoint = "SetProcessWorkingSetSize")]
public static extern int SetProcessWorkingSetSize(IntPtr process, int minSize, int maxSize);
/// <summary>    
/// 释放内存    
/// </summary>    
public static void ClearMemory()
{
    GC.Collect();
    GC.WaitForPendingFinalizers();
    if (Environment.OSVersion.Platform == PlatformID.Win32NT)
    {
        SetProcessWorkingSetSize(System.Diagnostics.Process.GetCurrentProcess().Handle, -1, -1);
    }
}

调用以上API能让你的内存一下爆减,是否是很给力,一调用内存就降下来了。But,先别高兴太早,这实际上是伪释放,只是暂时解决内存大量泄漏致使系统崩溃的应急处理方案。具体缘由参考:SetProcessWorkingSetSize函数的骗局,关键信息:物理内存转虚拟内存,涉及磁盘读写。好处坏处都贴出来了,是否须要使用请君本身斟酌。

4、总结

实际上因为各个开发人员的水平跟接触面不一样,又没有通过统一的培训(各我的对资源释放的理解与关注度不一样,或者写代码时就没考虑内存未被释放这种问题),发现问题的时候项目每每已经作到了一个阶段,系统也比较庞大了,这种时候才发现内存泄露的问题确实是很头疼的。

  • 资源泄露的场景每每是相互关联的,发生最多的就是内存泄漏,而除了写法可能有问题外,也多是由于句柄泄露或用户对象泄露引发的。

5、参考资料

相关文章
相关标签/搜索