Android系统调试(03)OOM问题总结(android oom解决方案)
haoteby 2025-06-23 19:12 13 浏览
1 内存泄露框架
@1 为什么会有内存泄漏?
一个不会被使用的对象,因为另一个正在使用的对象持有该对象的引用,导致它不能正常被回收,而停留在堆内存中,内存泄漏就产生了,Android系统为每个应用分配的内存是有限的,内存泄漏会使我们的应用内存随着时间不断的增加,造成应用OOM(Out Of Memory)错误,使应用崩溃。
@2 如何解决内存泄漏?
在解决内存泄漏的时候常常使用LeakCanary工具,它是一个自动检测内存泄漏的开源工具,使用它我们就可以明确的知道那个地方发生了泄漏。
2 造成内存泄露的几种情况
2.1 持有Context造成的内存泄漏
@1 原因:在Android中有两种context对象:Activity和Application.当我们给一个类传递context的时候经常使用第一种,而这样就导致该类持有对Activity的全部引用,当Activity关闭的时候因为被其他类持有,而导致无法正常被回收,从而而导致内存泄漏。
@2 解决方案:在给其他给传递context的时候使用Application对象,和这个对象的生命周期共存亡,而不依赖activity的生命周期,而对context的引用不要超过他本身的生命周期,谨慎对context使用static关键字。
2.2 Handler造成的内存泄漏
关键代码如下所示:
C++
public class SampleActivity extends Activity {
private final Handler mLeakyHandler = new Handler() {
@Override
public void handleMessage(Message msg) {
// ...
}
}
}
这样来使用Handler会造成严重的内存泄漏。
@1 原因:假设Hanlder中有延迟的任务或是等在执行的任务队列过长,由于消息队列持有对handler的引用,而handler又持有activity的隐式引用,这个引用会保持到消息得到处理,而导致activity无法被垃圾回收器进行回收,而导致内存泄漏。
@2 解决方案:可以把Handler放到单独的类中,或者使用静态的内部类(静态内部类不会引用activity)避免泄漏。如果想要在handler内部去调用Activity中的资源,可以在Handler中使用弱引用的方式指向所在的Activity,使用static+WeakReference的方式断开handler与activity的关系,最终代码如下:
Java
public static class MyHandler extends Handler {
//声明一个弱引用对象
WeakReference<MainActivity> mReference;
MyHandler(MainActivity activity) {
//在构造器中传入Activity,创建弱引用对象
mReference = new WeakReference<MainActivity>(activity);
}
public void handleMessage(Message msg) {
//在使用activity之前先判空处理
if (mReference != null && mReference.get() != null) {
mReference.get().text.setText("hello word");
}
}
}
2.3 使用单利模式造成的内存泄漏
@1 在我们使用单利模式的时候如果使用不当也会造成内存泄漏。原因:单例模式的静态特征使得单例模式的生命周期和应用一样的长,这说明了当一个对象不需要使用了,而单例对象还存在该对象的引用,那么这个对象就不能正常的被回收,造成内存泄漏。
@2 解决方案如下:
Java
XXUtils.getInstance(this);
这句代码默认传入的是Activity的Context,而Activity是间接继承自Context的,当Activity退出之后,单例对象还持有它的引用,所以在为了避免传Activity的Context,在单例中通过传入的context获取到全局的上下文对象,而不使用Activity的Context就解决了这个问题
Java
public class XXUtils {
private Context mContext;
private XXUtils(Context context) {
mContext = context.getApplicationContext();
}
private static XXUtils instance;
public static XXUtils getInstance(Context context) {
if (instance == null) {
synchronized (XXUtils.class) {
if (instance == null) {
instance = new XXUtils(context);
}
}
}
return instance;
}
}
2.4 非静态内部类创建静态实例造成的内存泄漏
@1 在项目中我们为了避免多次的初始化资源,常常会使用静态对象去保存这些对象,这种情况也很容易引发内存泄漏,原因如下:
非静态的内部类默认会持有外部类的引用
而我们又使用非静态内部类创建了一个静态的实例
该静态实例的声明周期和应用一样长,这就导致了该静态实例一直会持有该Activity的引用,导致Activity不能正常回收
@2 解决方案:
将内部类修改成静态的,这样它对外部类就没有引用,将该对象抽取出来封装成一个单例,最终代码:
Java
private static TestResource mTestResource;
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
findViewById(R.id.btn).setOnClickListener(this);
}
private void initData() {
if (mTestResource == null) {
mTestResource = new TestResource();
}
}
public void onClick(View v) {
initData();
}
//非静态内部类默认会持有外部类的引用
//修改就可以正常被回收,是因为静态的内部类不会对Activity有引用
private static class TestResource {
}
2.5 线程造成的内存泄漏
@1 原因:在使用线程时,一般都使用匿名内部类,而匿名内部类会对外部类持有默认的引用,当Acticity关闭之后如果现成中的任务还没有执行完毕,就会导致Activity不能正常回收,造成内存泄漏。
@2 解决方案:创建一个静态的类,实现Runnable方法,在使用的时候实例化他。最终代码:
Java
private void loadData() {
new Thread(new MyThread()).start();
}
private static class MyThread implements Runnable {
public void run() {
SystemClock.sleep(20000);
}
}
资源未关闭造成的内存泄漏。对于使用了BraodcastReceiver,ContentObserver,File,Cursor,Stream,Bitmap等资源的代码,应该在Activity销毁时及时关闭或者注销,否则这些资源将不会被回收,造成内存泄漏。
2.6 监听器没有注销造成的内存泄漏
@1 原因:在Android程序里面存在很多需要register与unregister的监听器,忘记unregister会导致内存泄露
@2 解决方案:我们需要确保及时unregister监听器。
2.7 集合中的内存泄漏
我们通常把一些对象的引用加入到了集合容器(比如ArrayList)中,当我们不需要该对象时,并没有把它的引用从集合中清理掉,这样这个集合就会越来越大。如果这个集合是static的话,那情况就更严重了。所以要在退出程序之前,将集合里的东西clear,然后置为null,再退出程序。
相关推荐
- 谷歌开源大模型评测框架正式发布,AI模型评测难题迎刃而解
-
近日科技巨头谷歌正式推出其开源大模型评测框架LMEval,这一创新工具为全球AI开发者和企业提供了标准化的模型评估解决方案。LMEval的发布不仅标志着AI模型评测迈入透明化时代,更通过多项核心技术...
- Android 开发中文引导-动画和图形概述
-
安卓系统提供了各种强大的API,用来将动画应用于界面元素和自定义2D和3D图形的绘制当中。下面的小节大概的描述了可用的API和系统功能并帮助你决定那个方案最适合你的需要。动画安卓框架提供了两种动画系统...
- Qt5 C++入门教程-第12章 绘图(QPainter)
-
QPainter类在Qt5中进行绘图时起着重要作用。绘图操作是通过QPainter类在响应paintEvent方法时完成的。线条在第一个示例中,我们在窗口的客户区绘制了一些线条。line...
- 文创测评︱《如意琳琅图籍》:本土原创解谜书的胜利?
-
设想这样一个场景,你打开一本书,就化身为乾隆三十六年紫禁城中的画画人周本,有一天你在故纸堆中找到一本神秘的《如意琳琅图籍》,踏上寻宝旅程,历经各种离奇复杂的故事……这是故宫与奥秘之家联手打造的创意解谜...
- gif动图制作攻略!快快收藏(求gif制作的动图)
-
有事没事斗图玩是当下人们乐此不疲的事情,手里的gif动图也渐渐成为了人们抬杠互怼的一大资本。好有趣,好炫酷,gif是怎么做出来的?我也想做。什么?你不会?没关系,我来教你!首先介绍一下制作gif动图需...
- eduis未能初始化界面 无法启动 问题解决办法
-
1.如果edius安装后启动后出现failedtoinitializeskin中文提示无法初始化界面的错误。这说明你的电脑安装了双显卡,而edius所使用的是图形显卡。可以选择edius图标右键...
- Flash Player模拟器更新:Rufffle(flash模拟器安卓下载高版本)
-
Ruffle是一个适用于WindowsPC的FlashPlayer模拟器,用Rust编写。Ruffle作为一个独立的应用程序在所有现代操作系统上原生运行,并通过使用WebAssembly在所有现代...
- 支持终身免费4G流量,星星充电7kW星际智能交流充电桩拆解
-
前言近期星星充电推出了一款星际智能交流充电桩,在正面设有灯条,可根据灯条颜色和显示直观了解充电状态,并设有屏幕显示充电状态和ui表情。充电桩支持220V/7kW充电功率,适配主流新能源车型。并支持终身...
- 乐动随心之fancy pop(乐动随心壶多少钱一个)
-
跳动飞扬的音符像是连通人与人之间心电感应的通关密码,融化陌生,拉近彼此。此次我们邀请到宅男女神江语晨,化身音乐精灵。在歌手、演员身份间游刃自如的她,为我们生动诠释了三种不同的音乐时尚风格,娴静可爱,灵...
- Asus Zenflash 手机也能玩引闪,从此相机是路人
-
在讲解Zenflash之前,不得不提索爱的K750c,这个机器采用了氙气闪光灯,让手机的拍摄上了档次,可玩性更高,不过,说实话,当时手机的摄像头像素低,成像一般,没有掀起太大的波澜,可现在,手机的Cm...
- Axure有哪些鲜为人知的使用技巧?(axure的使用教程)
-
阿拓带你飞:不管是想入门产品经理还是已经是PM的人对AXURE都很关注,它是制作产品原型的重要工具,但是有多少人了解AXURE的使用技巧?本文是来自“知乎问答”整理的回答,一起来看看那些不常用的使用技...
- 挑战黑夜 华硕ZenFlash氙气闪光灯评测
-
【机锋配件】说到摄影,相信许多朋友都非常喜欢,不管是外出游玩拍拍风景,还是和朋友之间聚会,都会掏出手机拍两张,在餐前拍照晒朋友圈更是成为了许多用户的日常爱好,就算不是专业的摄影爱好者,大家也都有一颗热...
- WPS 演示倒计时 3 步设置!从数字动画到进度条全场景教程
-
做PPT时想添加倒计时却找不到入口?WPS演示自带的"动画+计时"功能就能轻松实现——无论是课堂互动的30秒答题倒计时、商务汇报的5分钟限时讲解,还是活动暖场的动...
- flash动画an制作MG动画元素如何调节透明度,小白...
-
如何在flash动画软件里面调节mg动画元素的透明?因为flash动画软件现在已经升级为flash动画软件,所以直接用新版flash动画软件开工,基本功能都差不多,只是flash增加很多智能化、人性...