WPF不明内存泄露已解决,白头发也没了
在使用OpenExpressApp进行WPF应用开发过程中遇到多个内存泄漏的地方,在上一篇中求助了一个内存泄露问题【WPF不明内存泄露原因,头发都白了几根】,本篇与大家分享一下如何解决此问题的过程。
问题发现
用户使用产品时,使用久了会报出内存溢出错误,于是开始查找内存泄露问题。在【WPF -.Net 4.0解决了DataGrid分组时的内存泄露】中介绍了一个DataGrid内存泄露问题,这里主要说的是关闭模块后对象仍没有释放的问题。
问题解决
这个问题在本周之前已经由另一同事使用ANTS Memory Profiler 5查看定位到是Button导致,但是原因不明。由于内存泄漏不是小问题,所以我决定这周看看。
继续查找原因
使用ANTS Memory Profiler 5查看内存泄露问题,找到问题是在Button上,但是搞不清为什么会在button上,特别是加上了WPF的DependencyObject的一些东西,如果想要弄明白觉得还不简单。周一开始有时间我就对这个问题进行定位,但是到今天仍旧没有头绪:( 于是开始找其他办法。

- 使用其它内存泄露工具看看
网上google了一下内存泄露工具,随便点击一个,下载了Scitech memory profiler试用版本,新建项目后跑了一下程序,关闭模块后拍了一个快照,在Type Details页签输入ProjectList类库,显示下图,其路径和上图路径是一样的:
但是这个工具好在可以看到调用堆栈(可能ANTS Memory Profiler也有这个功能,只是我不知道),切换几个类查看堆栈,找到了在OpenExpressApp中的类库,高兴阿,如下图:
从上图可以看出可能是在ButtonCommand.SetCommand中代码导致泄漏的。 - 今天上午发帖【WPF -.Net 4.0解决了DataGrid分组时的内存泄露】
很感谢很多朋友马上给我回复,其中李永京肯定是有经验的高手,通过我的一个图就能猜出问题所在,我想以前他也一定受到这个的困扰。他的回复是这样的:
LZ 是不是用DependencyPropertyDescriptor.FromProperty
这样得到属性然后用 AddValueChanged加了委托跟踪属性值的变化,最后在unload的时候去掉下。
这个是猜测,没有具体代码也很难分析。
解决办法
通过上面的查找,基本定位可能出现在跟踪属性值变化的AddValueChanged中,查看项目代码,发现的确使用了
于是google搜索【PropertyDescriptor AddValueChanged leak】,看到了一篇于此相关的blog:PropertyDescriptor AddValueChanged Alternative。看了一下这篇文章,知道是由于事件强引用导致,blog中也提出了一个解决办法,就是建立一个PropertyChangeNotifier类,通过弱引用对象建立属性值更改事件的绑定,具体代码和说明参考这篇bolg即可PropertyDescriptor AddValueChanged Alternative,我就不再详述了。
问题定位了,解决方法也有了,于是开始动刀修改问题:全文搜索一下 DependencyPropertyDescriptor.FromProperty,发现有两处使用了,都是OpenExpressApp中引用的外部代码,一个是封装Command的ButtonCommand,一个限制ListView宽度的ViewLayoutManager,代码修改如下所示:(大家也可以通过WeakEventManager来解决)
//memory leak, use PropertyChangeNotifier
//DependencyPropertyDescriptor.FromProperty(
// Button.IsEnabledProperty,
// typeof(Button)).AddValueChanged(button, ButtonIsEnabledChanged);
PropertyChangeNotifier notifier = new PropertyChangeNotifier(button, "IsEnabled");
notifier.ValueChanged += new EventHandler(ButtonIsEnabledChanged);
代码修改后再跑测试,发现已经找不到这个对象了,终于解决了这个问题了:) 开始着手寻找其它内存泄漏问题了,不过现在有Scitech memory profiler应该能够更快定位了。
回顾
- 事件强引用是.Net下内存泄漏的常出现的由于编码不注意导致的问题
|
Leak Description |
Developer Error |
|
Improper Use of Event Handlers |
X |
|
Improper Use of Data Binding |
X |
|
Improper Use of Command Binding |
X |
|
Improper Use of Static Event Handlers |
X |
- 设定任务timebox,到时还未解决,可以尝试其它工具方法:例如使用其它工具;寻求有经验的热心人帮忙
相关blog
WPF -.Net 4.0解决了DataGrid分组时的内存泄露
欢迎转载,转载请注明:转载自周金根 [ http://zhoujg.cnblogs.com/ ]
WPF不明内存泄露已解决,白头发也没了的更多相关文章
- PF不明内存泄露已解决,白头发也没了(转)
在使用OpenExpressApp进行WPF应用开发过程中遇到多个内存泄漏的地方,在上一篇中求助了一个内存泄露问题[WPF不明内存泄露原因,头发都白了几根],本篇与大家分享一下如何解决此问题的过程. ...
- 分享.net常见的内存泄露及解决方法
分享.net常见的内存泄露及解决方法 关于内存泄漏的问题,之前也为大家介绍过,比如:<C++中内存泄漏的检测方法介绍>,是关于C++内存泄漏的.今天为大家介绍的是关于.NET内存泄漏的问题 ...
- JavaScript内存泄露,闭包内存泄露如何解决
本文原链接:https://cloud.tencent.com/developer/article/1340979 JavaScript 内存泄露的4种方式及如何避免 简介 什么是内存泄露? Java ...
- JS高程中的垃圾回收机制与常见内存泄露的解决方法
起因是因为想了解闭包的内存泄露机制,然后想起<js高级程序设计>中有关于垃圾回收机制的解析,之前没有很懂,过一年回头再看就懂了,写篇博客与大家分享一下. #内存的生命周期: 分配你所需要的 ...
- 调用MediaScannerConnection 发生内存泄露的解决方法
调用MediaScannerConnection发起扫描时经常会发生内存泄露,例如: E ActivityThread: Activity FolderListActivity has leaked ...
- 【javascript】内存泄露及其解决办法
1.内存泄露:一般由于开发者使用不当导致不用的内存没有被操作系统或者空闲内存池回收释放. 2.造成内存泄露的常见原因: 1) 意外的全局变量引起的内存泄露 2)闭包引起的内存泄露 闭包可以维持函数内局 ...
- Android APP常见的5类内存泄露及解决方法
1.static变量引起的内存泄漏 因为static变量的生命周期是在类加载时开始 类卸载时结束,也就是说static变量是在程序进程死亡时才释放,如果在static变量中 引用了Activity 那 ...
- 使用SDWebImage加载大量图片后造成内存泄露的解决办法
SDWebImage的知名度就不用说了,github上近10k的star,国内外太多的App使用其进行图片加载. 但是最近在使用过程中发现,在UITableView中不断加载更多的内容,使用SDWeb ...
- WPF常见内存泄露
Event handlers leak This type of leak occurs when subscribing an object (let's call it listener) to ...
随机推荐
- AE属性表操作
转自chanyinhelv原文AE属性表操作 实现的操作包括:1.打开属性表:2.编辑属性表:3.增加属性列:4.数据排序:5.字段计算…… 嗯,实现的功能目前就这些吧,后续还会继续跟进,还望大家多多 ...
- C#反射应用
考虑这个是因为返回的是对象集合,需要把对象集合绑定到datagridview上,绑定datagridview需要数据源,组装数据的话,用datatable添加列很麻烦,所以用反射来实现,估计可能会有多 ...
- 记录一次对接XX支付SDK过程中报错问题
我们支付平台以前我不做对接上游的,偶然间替别人做"对接了XX支付的相关接口的工作".在工作过程中发现SDK和对外提供服务过程中很容易出问题.在此做个记录,为了以后相关工作中作为自己 ...
- 关于用什么作为dll版本号的思考
作者:朱金灿 来源:http://blog.csdn.net/clever101 一个软件模块的版本如何维护呢?毫无疑问,它需要一个版本号.通过比对版本号就知道哪个高版本,哪个是低版本了.软件模块以d ...
- 【CTO俱乐部研修班开课】看板先驱David J. Anderson:看板核心在于创造一种能力——提升敏捷性
看板开发方法是近年来最热门的敏捷和精益开发方法.看板之父David J. Anderson觉得其核心在于帮助企业创造一种能力--提升敏捷性.CTO俱乐部看板研修班将通过理论.沙盘模拟.真实案例分享等阐 ...
- oracle 全部查询和表空间,以及其关系
select * from dba_users; 查看数据库里面全部用户,前提是你是有dba权限的帐号.如sys,system select * from all_users; 查看你能管 ...
- cordova通过指纹插件进行指纹验证
原文:cordova通过指纹插件进行指纹验证 版权声明:本文为博主原创文章,转载须注明出处,博客地址:https://blog.csdn.net/wx13227855087 https://blog. ...
- Linux常用 bash
学会Linux常用 bash命令 目录 基本操作1.1. 文件操作1.2. 文本操作1.3. 目录操作1.4. SSH, 系统信息 & 网络操作 基本 Shell 编程2.1. 变量2.2. ...
- Eclipse使用技巧总结(五)
三十五.快速回退到上次编辑处 Ctrl + Q 三十六.查看调用层次 Ctrl + Alt + H 三十七.快速查看某个类 Ctrl + Shift + T 三十八.快速定位 Ctrl + K :首 ...
- Qtcreator中常用快捷键总结(有必要牢记,提高编程效率)
F1 查看帮助F2 跳转到函数定义(和Ctrl+鼠标左键一样的效果)Shift+F2 声明和定义之间切换F4 头文件和源文件之间切换Ctrl+1 ...