- 浏览: 137834 次
- 性别:
- 来自: 郑州
文章分类
最新评论
-
Ken艹小哲:
太赞了 哥们 加扣
猜数字问题的最少步数算法. -
chenyuan122912:
...
解决AS3 Socket编程中最令人头疼的问题 -
hyx0914:
有源代码打包下载的吗?
Flex地图分块加载的实现 -
tianhai110:
非常有价值的 测试总结
时间效率,Timer和EnterFrame在FP 10.1之后测试和建议 -
763675117:
你好,fengxiangpiao,看了你的文档,我的需求和你差 ...
解决MovieClip打印问题 转
Flex 采用 ActionScript 语言作为脚本语言,编译后的二进制代码在 FlashPlayer 虚拟机 AVM(Actionscript Virtual Machine)中运行。和 Java 语言类似,AVM 中也有一个垃圾收集器(Garbage Collection),对于不用的对象,隔一段时间会进行收集并销毁,释放内存。和 C++ 语言相比,程序员不需要时刻关注内存的分配和释放,大大减轻了负担。但是垃圾收集器不能从根本上解决 Flex 内存泄露的问题,这得从 FlashPlayer 虚拟机的垃圾回收机制谈起。
垃圾收集器采用计数法或标记法来查找需要清除的对象。计数法由于无法检测循环引用的对象,现在已经很少采用了。重点谈一下标记法。Flex 应用的对象在内存中被映射成树形结构。这很好理解,每个 Flex 应用总有一个 Application 的入口被称为根节点(Root),垃圾收集器从根节点开始遍历每个对象,对可达对象标记为“有效”(有一种例外就是弱引用,后面的章节详谈)。而在这棵树之外的孤岛对象或者由于循环引用形成的孤岛对象集合被标记为“无效”,垃圾收集器会在合适的时间销毁这些无效对象,完成一次垃圾收集。而垃圾收集器是运行在虚拟机中的一个低优先级的守护进程,为了不影响性能,它只在必要的时候才运行。例如在向操作系统申请新内存空间的时候,发生异常的时候等等,因此内存并不是实时回收的。
有了垃圾收集器,为什么 Flex 还会产生内存泄露呢?从垃圾收集器的角度看,对象分为“有效”和“无效”两类;而从 Flex 应用程序的角度看,对象又被分为“有用”和“无用”两类。
举个例子,当程序出现逻辑错误需要提示用户时,Flex 程序构造一个提示框,这时,提示框是一个“有用”的对象,当用户点击关闭按钮关掉提示框后,提示框就变成“无用”的对象了,应用程序再也不会用到它(下次出现相同逻辑错误时,程序又会构造一个全新的提示框)。应用程序认为这个提示框应该被回收掉,但是因为某种原因,存在一个从“有效”对象到这个提示框的引用,垃圾收集器显然认为提示框也是“有效”的。这个“有效”“无用”的提示框便造成了 Flex 的内存泄露。
了解了 Flex 内存泄露的原因,从程序员的角度来讲,对于对象引用的混乱管理是造成 Flex 内存泄露的人为因素。Flex 开发中对于对象的引用分为两种:显示引用和隐式引用,我们分别就这两种情况讨论一下它们是如何造成内存泄露的。
- 表达式 b=a,创建一个从 b 指向 a 的引用,当 a 变成无用对象时,由于还存在 b 对它的引用,所以 a 的内存不能被回收。在开发过程中,全局变量、静态变量、特别是采用单例模式创建的对象,对其他对象的引用,如果不及时释放都极易造成内存泄露。例如:
清单 1. 表达式显示引用public static var staticVar : Object = new Object(); public function leak():void{ var chart : AreaChart = new AreaChart(); staticVar = chart; chart = null; }
在 leak()方法中,创建了一个临时变量 chart,然后将它赋给静态变量 staticVar,虽然最后将 chart 置为 null,但是由于静态变量对它有一个引用,chart 所占的内存不会被回收,造成内存泄露。
- 以对象为参数的方法,在方法体内部创建了指向该对象的引用,没有及时释放而导致内存泄露。将上面的代码变化一下:
清单 2. 以对象为参数的方法var chart : AreaChart = new AreaChart(); leak(chart); chart = null; ...... public static var staticVar : Object = new Object(); public function leak(chart : AreaChart):void{ staticVar = chart; }
原因和上例相同,只是发生的位置更加隐蔽。
- 对象自身的无参数方法调用,在方法体中创建对“this”关键字的引用,没有及时释放而导致内存泄露。下面这段程序是 UIComponent 的 setFocus() 的源代码:
清单 3.UIComponent.aspublic function setFocus():void { var sm:ISystemManager = systemManager; if (sm && (sm.stage || sm.useSWFBridge())) { if (UIComponentGlobals.callLaterDispatcherCount == 0) { sm.stage.focus = this; UIComponentGlobals.nextFocusObject = null; } ...
在调用 setFocus() 方法后,通过 sm.stage.focus = this,全局对象 systemManager 产生了对 UIComponent 的引用。如果之后不做处理,就会造成内存泄露。这里只是举个例子,对于 UIComponent 的 setFocus() 方法 Flex 已经做了处理,不会造成内存泄露,大家可以放心使用。但是在日常的编程过程中,一定要注意这种非常隐蔽的情况。
隐式引用最常见的情况就是添加事件监听器。a.addEventListener("Leak", b.leakHandler);a 对象的“listener”属性创建了一个指向 b 对象 leakHandler 方法的引用,如下图所示,
即使把 b 置为 null,只要 a 对象没有被回收,b 也不会被回收,从而导致内存泄露。通过弱引用方式可以避免这种内存泄露,在后面的章节会有详细描述。
Adobe 公司在 Flex Builder 3 中提供了一个 Profiler 工具,用于 Flex 内存诊断和性能调优。本文重点介绍 Profiler 用于内存诊断方面的功能。
在 Flex 开发视图,选择主入口文件并点击鼠标右键 ->Profile As->Flex Application 启动 Profiler 工具。如下图所示:
启动后,系统首先弹出对话框让用户配置 Profiler 的参数,如下图所示:
- 选项“Enable performance profiling”是用做性能调优的,主要用来找到响应时间的瓶颈。在做内存调试时,将这一项勾掉。
- 选项“Generate object allocation stack traces”选项可以跟踪对象创建的整个过程,这个功能非常消耗系统资源,在调试的初期,目的是找到内存泄露的对象,而不关心它的创建过程,因此先不要选择该项。
配置完毕,点击 Resume 按钮继续执行。
几秒钟之后,Profiler 工具开始运行,如下图所示:
“Profile”窗口显示正在运行的 SWF 应用,选中后,窗口上的一系列的按钮就变成可用状态,如下图所示:
这里简要介绍一下和内存调试相关的按钮的功能:
图 5 中 1“Run Garbage Collector”,点击该按钮,会强制执行一次内存回收。
图 5 中 2“Take Memory Sanpshot”,在运行的任何时刻,点击该按钮,系统首先会自动执行一次强制内存回收,然后捕获一帧内存快照,作为当前正在运行的应用的子对象,如下图所示:
双击这帧内存快照,会在下部窗口打开一个页签,显示该内存快照中的对象,包括对象所在的包,实例数目和百分比,所占内存和百分比,如下图所示:
图 5 中 3“Find Loitering Objects”,找出“游荡对象”。该按钮需要同时选中两帧内存快照(按住 Ctrl 键点选内存快照)才有用,如下图所示:
它的作用是通过对比两帧内存快照,找出在后一帧内存快照中存在而在前一帧内存快照中 不存在的对象。在某些特定的场景中,该功能能够迅速找到内存泄露的对象。
Flex 应用内存泄露的最直观表象是当用户进行某些相同操作时,内存和对象实例会持续增加,即使进行了垃圾回收,内存也不会回到原始的水平。
在上一章节中,我们了解了 Profiler 的主要功能,它可以在任何时刻捕获当时的内存快照。那么可以想象,如果每当用户进行一次相同的操作时,我们就捕获一帧内存快照,通过对比几帧快照,找出持续增加的对象实例,就可以发现是哪些对象导致了内存泄露,从而发现程序中的漏洞,最终解决问题。
我们通过一个例子来实践一下这个假设。本文提供了一个模拟生成验证码的 Flex 程序 DetectMemoryLeak.mxml,如下图所示:
每当用户点击“Change”按钮时,就会随机产生一个 4 位数字替换原来的数字。该程序在用户每次点击按钮后都会导致内存的增加,我们怀疑这个程序存在内存泄露。启动 Profiler 工具,用户每点击一次“Change”按钮,我们就捕获一帧内存快照,连续做 3 次,我们对比一下这 3 帧快照,如下图所示:
通过对比,我们发现 NumberChangeLabel 的实例数目随着用户的每次点击都会增加,并且不会被回收,因此断定是 NumberChangeLabel 导致了内存泄露。双击最后一帧内存快照中的 NumberChangeLabel 对象,会打开“Object References”页签,显示这 4 个实例被哪些对象引用,如下图所示:
表中的 4 条数据代表这帧内存快照中有 4 个 NumberChangeLabel 实例,每个实例后面括号中的数字,比如(7),代表了有多少个引用指向这个实例。点击“+”号,逐项展开,如下图所示:
我们发现绝大部分是对象自身或者对象的子元素对该对象的引用,这种引用不会产生内存泄露。而有一项 DetectmemoryLeak 的 arr 属性对 NumberChangeLabel 的引用是外部引用,也正是因为这个引用导致 NumberChangeLabel 不能被回收。检查相关的源代码:
private static var arr : Array = new Array(); private function changeClickHandler(event:MouseEvent):void{ labelContainer.removeAllChildren(); var label : NumberChangeLabel = new NumberChangeLabel(); labelContainer.addChild(label); arr.push(label); } |
发现 arr 是静态数组,arr 的 push 操作导致了内存泄露,去掉 arr.push(label) 后,问题解决。至此,内存泄露的诊断就完成了。
内存快照对比法虽然直观,但是当应用非常复杂时,可能有成百上千个对象,仅通过肉眼对比,效率和准确率会大大降低。我们想到了 Profiler 工具中的“Find Loitoring Objects”查找游荡对象功能,可不可利用这个功能来找出内存泄露的对象呢?从“游荡”对象的定义来看,它是在后一帧内存中存在而前一帧内存中不存在的对象,那么游荡对象是不是就是导致内存泄露的对象呢?对于上个模拟生成验证码的例子,经过修改,程序不再存在内存泄露,当用户点击“Change”按钮时,产生一副新图片 image1,替换前一帧内存中的旧图片 image0,image0 将被垃圾收集器回收。根据定义“image1”是游荡对象,很明显它不会导致内存泄露。
那么游荡对象有什么用处呢?假设有这样一个场景,用户从初始状态 A 进行了一系列操作到达终止状态 B,A 和 B 的状态完全相同,就好像又回到初始状态一样。举个例子,用户从初始状态开始,点击按钮弹出对话框,再关闭对话框后,到达终止状态,起始状态和终止状态完全相同,那么此时如果存在游荡对象就会导致内存泄露了。
运行本文提供的示例程序 LoiteringMemoryLeak.mxml 来模拟这个场景,初始状态只有一个按钮“Popup a dialogue”, 如下图所示:
点击“Popup a dialogue”弹出对话框,如下图:
点击“OK”按钮关闭对话框之后,到达终止状态。在初始状态和终止状态各捕获一帧内存快照,选中两帧快照点击“Finding Loitoring Objects”按钮,打开“Loitoring Objects”页签,如下图:
产生了很多游荡对象,别担心,大部分对象都是由一个或几个高层对象产生的,找出产生内存泄露的高层对象就能解决问题。我们发现游荡对象“LeakDialogue”是程序中的一个用户自定义对话框,它可能是产生了内存泄露的根源。双击这个条目打开“Object Reference”页签,如下图:
点击“+”号逐项展开,通过分析发现,LeakDialogue 的 leakHandler 方法被全局对象 systemManager 的 listener21 引用导致了内存泄露,检查程序发现问题果然出在 systemManager 的事件监听环节:
private function changeClickHandler(event:MouseEvent):void{ var ld : LeakDialogue = new LeakDialogue(); PopUpManager.addPopUp(ld, this, true); PopUpManager.centerPopUp(ld); systemManager.addEventListener(KeyboardEvent.KEY_DOWN, ld.leakHandler); } |
去掉最后一句,问题解决。
需要说明一点,读者可能发现即使改正后,还是会有少量的游荡对象,这是因为用户的动作会触发事件,Flex 会为事件和事件处理创建内部对象,这些对象不会引发内存泄露,因此只需要关注有没有高层控件成为游荡对象就可以了。
我们暂且把需要回收的对象称为临时对象,那么从根本上讲,对于临时对象引用的管理不当是引发内存泄露的根本原因,以下是几点关于如何避免内存泄露的建议。
- 对于显示引用,要尽量减少对临时对象的引用,尤其是全局变量,静态变量,使用单例模式创建的变量对临时变量的引用。这些变量包含 stage,systemManager,application, MVC 框架中 Model 和 Controller,还有以 Manager 命名的对象等等。另外,临时变量本身要尽量做到高内聚性,对象内部尽量减少对外部对象尤其是全局对象的依赖。
- 对于隐式引用,使用弱引用方式注册事件监听器,将最后一个参数 useWeakReference 设置为 true:a.addEventListener("Leak", b.leakHandler, false, 0, true); 这样做的结果是垃圾回收器在做标记时,会忽略 a 对于 b 的引用,如果 b 没有被其他对象引用,垃圾回收器就把它标记为“无效”进而回收,从而避免内存泄露,如下图所示:
另外,自引用方式 a.addEventListener("Leak", this.leakHandler); 子对象对父对象的引用方式 children.addEventListener("Leak", parent.leakHandler); 都不会产生内存泄露。
- 对于自定义组件要尽量早地做内存泄露的测试。如果等到集成之后再做整体的测试,系统的复杂度变高,组件的层次和关联过于复杂,要定位到问题的根源,是一件非常费时费力的事。
- Flex 还是一门比较新的技术,Flex 框架本身也存在一些内存泄露的问题。好在 Adobe 已经将它的 Bug 系统公开,开发人员可以到 http://bugs.adobe.com/去查找已经存在的内存泄露问题和一些官方的解决方案,相信对解决内存泄露问题会有所帮助。
本文从 Flashplayer 的垃圾回收机制谈起,分析了 Flex 产生内存泄露的原因,以及开发中常见的内存泄露的场景,介绍了如何利用 Flex Builder 自带的 Profiler 工具分析和诊断内存泄露,最后根据实际经验给出了一些避免内存泄露的建议。希望通过本文能给开发人员一些启示,开发出更健壮的 Flex 应用。
<!-- CMA ID: 555402 --><!-- Site ID: 10 --><!-- XSLT stylesheet used to transform this file: dw-article-6.0-beta.xsl -->
示例代码 | sourceCode.zip | 600KB | HTTP |
发表评论
-
最全的Flex网站
2011-10-15 23:18 732一、国外站点 1.资源 类Adobe Flex 2 ... -
时间效率,Timer和EnterFrame在FP 10.1之后测试和建议
2011-04-23 22:19 1059关于Timer还是EnterFrame的选择,一直是很 ... -
时间效率,Timer 和 EnterFrame 在 FP 10.1 之后测试和建议
2011-04-23 22:16 1062关于Timer还是EnterFrame的选择,一直是很多AS3 ... -
Socket粘包问题
2011-04-22 23:21 1032这些天做类似QQ空间的 ... -
关于Flash Socket通信的安全策略问题的一点心得
2011-04-22 22:36 701昨天做测试的时候遇到 ... -
FLEX入门实例--------FLEX全屏功能实现及右键菜单
2011-03-26 11:58 815FLEX的全屏功能的实现,光写AS还不够,还需要对JS部分做一 ... -
flex中的Text属性
2010-12-08 12:31 873不同于Lable只显示单行文本,Text可以显示多行文本, ... -
Flex跨域问题
2010-11-08 17:55 791From ESRI: The client brow ... -
Flex屏蔽并自定义鼠标右键菜单(转载)
2010-11-08 17:53 882最近手头有个项目需要屏蔽Flex鼠标右键菜单并自定义新的fle ... -
使用 IBM Rational Functional Tester 7.0.1 测试 Flex 应用程序
2010-11-05 21:45 842关于这篇文章必要的信息 这篇文章假定您已经对使用 IBM® ... -
在 Ajax 应用程序中实现实时数据推送
2010-11-05 21:40 670简介 Ajax 技术已经存在了一段时间,开发的动力已经真正开 ... -
使用 Flex 和 Dojo 开发交互式Web应用程序
2010-11-05 21:38 995引言 以用户体验为核 ... -
flex SystemManger
2010-11-05 16:35 692软件编程牛人网 Framework Fundamentals ... -
Flex 不同 application 之间传参数
2010-11-05 16:33 487软件编程牛人网 通过主影片来获得所有子影片的systemma ... -
如何在flash保留小数位
2010-11-04 20:07 826在家上网赚钱更容易 fl ... -
实例讲解Flex3的SharedObject对象用法
2010-11-04 16:01 759在家上网赚钱更容易 Flex中的SharedObj ... -
Flex中12个最基本、最简单但却最有用的函数
2010-11-04 15:54 692在家上网赚钱更容易 No.1 复制内容到剪贴板 S ... -
谈谈Flex中ArrayCollection的复制(克隆)
2010-11-04 15:51 725在家上网赚钱更容易 在Flex的开发过程中,常常会遇到复 ... -
flex中的拖放操作(一)
2010-11-04 15:47 934在家上网赚钱更容易 可视化的开发环境最大的特点是允许你在 ... -
flex中的拖放操作(二)
2010-11-04 15:46 773在家上网赚钱更容易 手动添加 drag-and-drop ...
相关推荐
内存泄露在Flex应用中通常由于以下两个主要原因: 1. 显示引用:开发者创建的显式引用,如全局变量、静态变量或单例模式下的对象引用,如果没有正确释放,会导致对象仍然被认为是“有效”的,即使它们在应用逻辑中...
### Flex应用内存泄露的分析与诊断 #### 引言 Flex是一种用于构建和部署富互联网应用(RIA)的技术,它采用ActionScript语言作为脚本语言,并在Flash Player虚拟机(AVM)中运行。如同其他高级编程语言一样,AVM...
Adobe Flex Profiler是一款功能强大的工具,可以帮助开发者识别Flex应用程序中的性能瓶颈或内存泄漏问题。通过本教程,我们将详细介绍如何利用Adobe Flex Profiler进行性能分析。 #### 一、Adobe Flex Profiler简介...
ExternalInterface用于与宿主环境(通常是浏览器)进行交互,而LocalConnection则用于Flex应用内部不同部分之间的通信。前者主要用于调用JavaScript函数,后者则用于在AS之间传递消息。 ### 29. 是否使用Flex的国际...
书中提供了从准生产环境到测试负载设计的性能调优策略,以及系统性诊断、JVM垃圾回收监控、线程Dump分析和Mission Control工具的使用技巧,帮助开发者高效地识别和解决性能瓶颈及内存泄露等问题。 综上所述,《Java...
开发者在使用NPSWF32调试版本时,可以通过专门的Flash Player Debugger工具(如FlashDevelop或Flex Builder)来查看错误日志,这有助于定位代码中的错误,特别是与ActionScript(Flash的编程语言)相关的逻辑错误。...
垃圾回收(Garbage Collection, GC)是自动内存管理的一种方式,主要用于处理程序中不再使用的对象,防止内存泄漏。Java、Python和.NET等编程语言都内置了垃圾回收机制。主要的垃圾回收方法有标记-清除、复制、标记-...