在使用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分组时的内存泄露

WPF不明内存泄露原因,头发都白了几根

PF不明内存泄露已解决,白头发也没了(转)的更多相关文章

  1. WPF不明内存泄露已解决,白头发也没了

    原文:WPF不明内存泄露已解决,白头发也没了 在使用OpenExpressApp进行WPF应用开发过程中遇到多个内存泄漏的地方,在上一篇中求助了一个内存泄露问题[WPF不明内存泄露原因,头发都白了几根 ...

  2. 分享.net常见的内存泄露及解决方法

    分享.net常见的内存泄露及解决方法 关于内存泄漏的问题,之前也为大家介绍过,比如:<C++中内存泄漏的检测方法介绍>,是关于C++内存泄漏的.今天为大家介绍的是关于.NET内存泄漏的问题 ...

  3. JavaScript内存泄露,闭包内存泄露如何解决

    本文原链接:https://cloud.tencent.com/developer/article/1340979 JavaScript 内存泄露的4种方式及如何避免 简介 什么是内存泄露? Java ...

  4. JS高程中的垃圾回收机制与常见内存泄露的解决方法

    起因是因为想了解闭包的内存泄露机制,然后想起<js高级程序设计>中有关于垃圾回收机制的解析,之前没有很懂,过一年回头再看就懂了,写篇博客与大家分享一下. #内存的生命周期: 分配你所需要的 ...

  5. 调用MediaScannerConnection 发生内存泄露的解决方法

    调用MediaScannerConnection发起扫描时经常会发生内存泄露,例如: E ActivityThread: Activity FolderListActivity has leaked ...

  6. 【javascript】内存泄露及其解决办法

    1.内存泄露:一般由于开发者使用不当导致不用的内存没有被操作系统或者空闲内存池回收释放. 2.造成内存泄露的常见原因: 1) 意外的全局变量引起的内存泄露 2)闭包引起的内存泄露 闭包可以维持函数内局 ...

  7. Android APP常见的5类内存泄露及解决方法

    1.static变量引起的内存泄漏 因为static变量的生命周期是在类加载时开始 类卸载时结束,也就是说static变量是在程序进程死亡时才释放,如果在static变量中 引用了Activity 那 ...

  8. 使用SDWebImage加载大量图片后造成内存泄露的解决办法

    SDWebImage的知名度就不用说了,github上近10k的star,国内外太多的App使用其进行图片加载. 但是最近在使用过程中发现,在UITableView中不断加载更多的内容,使用SDWeb ...

  9. Android中使用Handler造成内存泄露的分析和解决

    什么是内存泄露?Java使用有向图机制,通过GC自动检查内存中的对象(什么时候检查由虚拟机决定),如果GC发现一个或一组对象为不可到达状态,则将该对象从内存中回收.也就是说,一个对象不被任何引用所指向 ...

随机推荐

  1. mysql数据库优化课程---17、mysql索引优化

    mysql数据库优化课程---17.mysql索引优化 一.总结 一句话总结:一些字段可能会使索引失效,比如like,or等 1.check表监测的使用场景是什么? 视图 视图建立在两个表上, 删除了 ...

  2. 黑客落网记:FBI如何抓捕Anonymous核心成员

    腾讯科技讯 美国新闻网站Daily Dot近日撰文,通过他们掌握的资料和实地采访,还原了Anonymous核心成员被捕的经过. 以下为文章全文: 哈蒙德被捕前夜 2012年3月,一个周六的下午,天气异 ...

  3. ArcMap加载在线地图

    SimpleGIS 小小的SimpleGIS除了提供6大地图让人喜爱之外,更有其他的能耐同样让你爱不释手. 功能1:作为出图底图地图提供商中Bing.天地图两家提供的地图是无偏移的地图,所以可直接应用 ...

  4. flask学习(十一):if判断语句

    1. 语法: {% if xxx %} {% else %} {% endif %} 2. if的使用,和python中if相差无几 用if判断,当用户名存在且年龄大于18时,才显示用户名和注销

  5. AOP(面向切面)的粗俗理解

    百度百科的解释:AOP主要实现的目的是针对业务处理过程中的切面进行提取,它所面对的是处理过程中的某个步骤或阶段,以获得逻辑过程中各部分之间低耦合性的隔离效果. 一个比较绕的概念,简单来说就是把不影响业 ...

  6. Emacs Helm: 使用关键字搜索、获取、执行任何东西

    Helm 是一个emacs的软件包,定义了一个通用框架,交互式地.动态缩减式地使用关键字选择.获取.执行任何东西.比如: 执行emacs 命令 打开文件 查看man文档 执行grep操作 执行apt命 ...

  7. centos7 配置 NFS mount挂载服务器

    1. NFS服务端 安装NFS服务 yum install nfs_utils yum install rpcbind (系统默认已经有了,可查看下) 配置共享文件夹 1.    创建文件夹: mkd ...

  8. Git远程操作详解(新手必备)

    Git是目前最流行的版本管理系统,学会Git几乎成了开发者的必备技能. Git有很多优势,其中之一就是远程操作非常简便.本文详细介绍5个Git命令,它们的概念和用法,理解了这些内容,你就会完全掌握Gi ...

  9. 数据存储-- Core Data的使用(二)

    一.基础概念深入 1.NSManagedObjectContext 被管理数据上下文就像便笺簿 当从数据持久层获取数据时,相当于把这些临时的数据拷贝写在便笺簿上,然后就可以随心所欲的修改这些值. 通过 ...

  10. week13《java程序设计》第13次作业总结

    week13<java程序设计>第13次作业总结 1. 本周学习总结 以你喜欢的方式(思维导图.OneNote或其他)归纳总结多网络相关内容. 答: 1.IP与端口:ip和域名相对应,可找 ...