System.Net.NetworkInformation.Ping存在的“内存泄漏”问题

昨天写了一个监测网络状况的小工具,其实就是定时ping网关和一些地址。跑了一夜,今天发现电脑卡,昨天写的小工具占了4G多内存,明显内存泄露。小工具很简单,使用非托管资源的只有System.Net.NetworkInformation.Ping这个类。Ping调用的是异步方法SendAsync, 我在PingCompleted事件处理中已经Dispose了:

ContractedBlock.gif ExpandedBlockStart.gif Code
private void OnPingCompleted (object sender, PingCompletedEventArgs e)
{
    dot.gif
    (sender 
as Ping).Dispose();
}

于是调试,发现这句也确实执行了。不解,仔细研究代码。结果某次修改代码如下时,问题消失:

ContractedBlock.gif ExpandedBlockStart.gif Code
private void OnPingCompleted (object sender, PingCompletedEventArgs e)
{
    
using (sender as Ping)
    {
    dot.gif
    }
}

于是思考两段代码的区别,难道是偶尔发生异常导致Dispose没有执行?将Dispose放到开头:

ContractedBlock.gif ExpandedBlockStart.gif Code
private void OnPingCompleted (object sender, PingCompletedEventArgs e)
{
    (sender 
as Ping).Dispose ();
    dot.gif
}

问题依旧。显式调用IDisposable.Dispose(),问题解决:

ContractedBlock.gif ExpandedBlockStart.gif Code
private void OnPingCompleted (object sender, PingCompletedEventArgs e)
{
    (sender 
as IDisposable).Dispose ();
    dot.gif
}

上MSDN查,发现Ping.Dispose()并非实现了IDisposable.Dispose()方法,而是直接继承了Component.Dispose(),换言之,这个方法根本没有对Ping对象的非托管资源进行释放。反而Ping类显示实现了IDisposable接口,必须将其强转为IDisposable才能正确调用真正的Dispose()方法。

 

再看Component类,Component.Dispose()方法是一个非虚方法,而Component.Dispose(bool)是一个虚方法,Component.Dispose()方法只是对Component.Dispose(bool)的一个简单调用。按照微软的设计思想,Ping类应该重写Component.Dispose(bool)方法,实现资源释放,而不应显式实现IDisposable接口。因此,这应是一个bug。微软已声明将修复它,但从.NET 2.0开始,直到目前.NET 4.0 Beta 1经实验依然没有解决。

为避免遇到类似问题,再次强烈建议使用using语句块!

 

国外探讨此问题的两篇文章:
http://blog.mbcharbonneau.com/2006/11/14/using-the-ping-class-in-net-20-without-memory-leaks/
http://www.julmar.com/blog/mark/PermaLink,guid,fc83197b-ce2d-47ab-b70a-db352085b370.aspx

微软关于此bug的报告及回复:
http://connect.microsoft.com/VisualStudio/feedback/ViewFeedback.aspx?FeedbackID=98421

转载于:https://www.cnblogs.com/Gildor/archive/2009/09/08/1562225.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值