避免内存泄露
Android应用程序被限制在16MB的堆上运行,至少在T-Mobile G1上是这样。对于手机来说,这是很大的内存了;但对于一些开发人员来说,这算是较小的了。即使你不打算使用掉所有的内存,但是,你也应该尽可能少地使用内存,来确保其它应用程序得以运行。Android在内存中保留更多的应用程序,对于用户来说,程序间切换就能更快。作为我(英文作者)工作的一部分,我调查了Android应用程序的内存泄露问题,并发现这些内存泄露大多数都是由于相同的错误导致的,即:对Context拥有较长时间的引用。
在Android上,Context常用于许多操作,更多的时候是加载和访问资源。这就是为什么所有的Widget在它们的构造函数里接受一个Context的参数。在一个正常的Android应用程序里,你会看到两种Context类型,Activity和Application。而一般在需要一个Context的类和方法里,往往传入的是第一种:
@Override
protected void onCreate(Bundle state) {
super.onCreate(state);
TextView label = new TextView(this);
label.setText("Leaks are bad");
setContentView(label);
}
这意味着,View拥有对整个Activity的引用以及Activity自身拥有的所有内容;一般是整个的View层次和它的所有资源。因此,如果你“泄露”了Context(“泄露”指你保留了一个引用,阻止了GC的垃圾回收),你将泄露很多的内存。如果你不够仔细的话,很容易就能泄露一个Activity。
当屏幕的方向发生改变时,一般系统会销毁当前的Activity并创建一个新的,并保存它的状态。当系统这样做时,Android会从资源中重新加载应用程序的UI。假设你写的应用程序拥有大的位图,而你又不想在每次旋转时重新加载它。这里有最简单的方式,那就是在一个静态的字段里进行保存:
private static Drawable sBackground;
@Override
protected void onCreate(Bundle state) {
super.onCreate(state);
TextView label = new TextView(this);
label.setText("Leaks are bad");
if (sBackground == null) {
sBackground = getDrawable(R.drawable.large_bitmap);
}
label.setBackgroundDrawable(sBackground);
setContentView(label);
}
这段代码效率很快,但同时又是极其错误的;在第一次屏幕方向切换时它泄露了一开始创建的Activity。当一个Drawable附加到一个View上时,View会将其作为一个callback设定到Drawable上。上述的代码片段,意味着Drawable拥有一个TextView的引用,而TextView又拥有Activity(Context类型)的引用,换句话说,Drawable拥有了更多的对象引用(依赖于你的代码)。
这是最容易泄露Context的例子之一,你可以看看Home Screen源代码里是如何处理的(搜索unbindDrawables()方法):当Activity销毁时,设定存储的Drawable的callback为null。有趣的是,还有很多一连串的Context泄露情况,并且是非常糟糕的。这些情况会使得应用程序很快耗尽内存。
这里,有两种简单的方式可以避免与Context相关的内存泄露。最显而易见的一种方式是避免将Context超出它自己的范围。上面的例子代码给出的静态引用,还有内部类和它们对外部类的隐式引用也是很危险的。第二种解决方案是使用Application这种Context类型。这种Context拥有和应用程序一样长的生命周期,并且不依赖Activity的生命周期。如果你打算保存一个长时间的对象,并且其需要一个Context,记得使用Application对象。你可以通过调用Context.getApplicationContext()或Activity.getApplication()轻松得到Application对象。
概括一下,避免Context相关的内存泄露,记住以下事情:
- 不要保留对Context-Activity长时间的引用(对Activity的引用的时候,必须确保拥有和Activity一样的生命周期)
- 尝试使用Context-Application来替代Context-Activity
- 如果你不想控制内部类的生命周期,应避免在Activity中使用非静态的内部类,而应该使用静态的内部类,并在其中创建一个对Activity的弱引用。这种情况的解决办法是使用一个静态的内部类,其中拥有对外部类的WeakReference,如同ViewRoot和它的Winner类那样
- GC(垃圾回收)不能解决内存泄露问题
分享到:
相关推荐
### Java避免内存泄露的关键知识点 #### 一、内存泄露的概念及原因 内存泄露是指程序在申请内存后未能释放,导致这部分内存无法再次被利用。在Java中,由于具备垃圾回收机制(GC),理论上开发者不必担心内存泄露...
opencv3和opencv4多线程内存泄漏问题:以cv::resize函数测试结果为例。 使用中可修复或者可避免内存泄漏:1)使用opencv2的版本;2)在代码中设置修复该问题.
以下是避免Android内存泄漏的一些关键点和检查泄漏原因的方法。 首先,理解Android内存管理的基本原则至关重要。每个Android应用运行在一个独立的Linux进程中,当应用退出时,其分配的内存会随之释放。然而,如果...
Android webview 内存泄露的解决方法 最近在activity嵌套webview显示大量图文发现APP内存一直在涨,没法释放内存,查了很多资料,大概是webview的一...这样动态生成webview就能避免内存泄漏,可是这样会导致部分机型的w
本文将详细介绍这两个函数的使用方法和解决方案,帮助开发者避免内存泄露的问题。 一、cvLoadImage函数的内存泄露 cvLoadImage函数是OpenCV中加载图像的函数,但是如果不正确地使用该函数,将导致内存泄露。例如,...
应始终记住调用对应的释放函数(如`CloseHandle`)来避免内存泄漏。使用完毕后立即释放资源是良好的编程习惯。 例四:`PostMessage`可能导致的内存泄漏 在发送自定义消息时,如果消息结构体是动态分配的,接收方...
在探讨QT内存管理机制及如何避免内存泄露之前,我们首先应当理解何为内存泄露。内存泄露是指程序中已分配的堆内存由于未能释放,导致一直占用这部分内存空间,最终可能导致应用程序运行缓慢甚至崩溃。在C++中,内存...
静态内部类不会隐式地持有对外部类的引用,因此可以避免内存泄漏。但为了在静态内部类中访问外部类的成员,我们可以使用`WeakReference`。`WeakReference`只保持弱引用,当没有其他强引用指向被引用的对象时,该...
在C++编程中,内存管理是一项关键任务,而内存泄漏是开发者经常遇到...无论是通过静态分析、动态分析还是使用特定工具,都需要开发者对内存管理有深入的理解,并养成良好的编程习惯,这样才能避免内存泄漏带来的问题。
解决内存泄漏问题的方法是:定位那些有内存泄漏的类或对象并修改完善这些类以避免内存泄漏。方法是:经过一段时间的测试、监控,如果某个类的对象数目屡创新高,即使在 JVM Full GC 后仍然数目降不下来,这些对象...
此外,开发者还需要遵循良好的编程实践,比如合理使用静态变量、正确管理资源的生命周期等,从而避免内存泄漏的发生。 总之,解决Java内存泄漏的关键在于预防。通过提高代码质量、加强测试以及持续监控,可以在开发...
为了避免内存泄漏,开发者应: - 使用智能指针(如C++中的`std::shared_ptr`和`std::unique_ptr`)来自动管理内存。 - 定期进行代码审查和静态代码分析,查找潜在的内存泄漏点。 - 利用动态内存检测工具,如...
通过理解这些知识点,并结合适当的工具和编程习惯,开发者可以在VC++环境中有效地检查和避免内存泄露,确保程序的稳定性和高效性。在实际开发过程中,持续学习和实践这些方法,将有助于提升代码质量,降低维护成本。
了解Java的内存分配策略对于避免内存泄漏至关重要。Java程序在运行时,根据变量的类型和作用域,会在静态存储区(方法区)、栈区或堆区分配内存。栈内存用于存放基础类型变量和对象引用,当作用域结束时自动释放;堆...
MFC(Microsoft Foundation Classes)是微软提供的一个C++类库,用于简化Windows应用程序开发,其中CString是一...对于开发人员来说,理解这些知识点有助于他们在使用CString时避免内存泄露,提高程序的稳定性和效率。
如何高效避免Android内存泄露,文档时根据我多年的Android经验编写而成,希望对您有所帮助。
总之,虽然Java的垃圾回收机制为内存管理提供了一定的便利,但Android应用开发者仍然需要注意正确管理内存,以避免内存泄漏的发生。通过学习和实践上述知识点,开发者可以提高编写高效且稳定的应用程序的能力。
除了使用这些工具,程序员还可以通过良好的编程习惯来避免内存泄漏,例如: - 使用智能指针(如C++中的`unique_ptr`和`shared_ptr`)来自动管理内存。 - 避免全局变量和静态变量,它们可能导致内存泄漏更难以追踪。 ...
7. **优化建议**:除了基本的检测功能,高级的内存泄露监控工具可能还会提供优化建议,指导开发者如何改进代码以避免内存泄露。 文件名为"sIEve-0.0.8.exe",推测这可能是该内存泄露监控软件的可执行文件,版本号为...