Android OOM:内存管理分析和内存泄露原因总结

free§; // free memory

}

或者:

JNIEXPORT voidJNICALL Java_com_example_demo_TestMemory_nativeMalloc(JNIEnv *, jobject)

{

SLOGD(“allocate 50M Bytesmemory”);

char *p = new char[1024 * 1024 * 50];

if (p != NULL)

{

//memory will not usedwithout calling memset()

memset(p, 1, 1024102450);

} else SLOGE(“newobject failure.”);

free§; //free memory

}

malloc或者new申请的内存是虚拟内存,申请之后不会立即映射到物理内存,即不会占用RAM。只有调用memset使用内存后,虚拟内存才会真正映射到RAM。

9. 明明还有很多内存,但是发生OOM了。。
  • 这种情况经常出现在生成Bitmap的时候。

  • 在一个函数里生成一个13m 的int数组,再该函数结束后,按理说这个int数组应该已经被释放了,或者说可以释放,这个13M的空间应该可以空出来。

  • 这个时候如果你继续生成一个10M的int数组是没有问题的,反而生成一个4M的Bitmap就会跳出OOM。这个就奇怪了,为什么10M的int够空间,反而4M的Bitmap不够呢?

在Android中:

  1. 一个进程的内存可以由2个部分组成:java 使用内存 ,C 使用内存

这两个内存的和必须小于16M,不然就会出现大家熟悉的OOM,这个就是第一种OOM的情况。

  1. 一旦内存分配给Java后,以后这块内存即使释放后,也只能给Java的使用

这个估计跟java虚拟机里把内存分成好几块进行缓存的原因有关,反正C就别想用到这块的内存了,所以如果Java突然占用了一个大块内存,即使很快释放了:

  • C 能使用的内存 = 16M - Java某一瞬间占用的最大内存

  • Bitmap的生成是通过malloc进行内存分配的,占用的是C的内存,这个也就说明了,上述的4MBitmap无法生成的原因,因为在13M被Java用过后,剩下C能用的只有3M了。

二、了解dalvik的Garbage Collection

如图所示:

这里写图片描述

  • GC会选择一些它了解 还存活的对象 作为 内存遍历的根节点GC Roots),比方说thread stack中的变量JNI中的全局变量zygote中的对象(class loader加载)等,然后开始对heap进行遍历。到最后,部分没有直接或者间接引用到GC Roots的就是需要回收的垃圾,会被GC回收掉

如下图蓝色部分。

这里写图片描述

三、常见的内存泄漏

1. 非静态内部类 的静态实例 容易造成内存泄漏

public class MainActivity extends Activity

{

// 非静态内部类的静态实例

static Demo sInstance = null;

@Override

public void onCreate(BundlesavedInstanceState)

{

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

if (sInstance == null) {

sInstance= new Demo();

}

}

class Demo

{

void doSomething()

{

System.out.print(“dosth.”);

}

}

}

  • 上面的代码中的 sInstance 实例 类型为静态实例,在第一个MainActivity act1实例创建时,sInstance会获得并一直持有act1的引用。

  • 当MainAcitivity销毁后重建,因为sInstance持有act1的引用,所以act1是无法被GC回收的,进程中会存在2个MainActivity实例(act1和重建后的MainActivity实例),这个act1对象就是一个无用的但一直占用内存的对象,即无法回收的垃圾对象。

  • 所以,对于lauchMode不是singleInstance的Activity, 应该避免在activity里面实例化其非静态内部类的静态实例

2. Activity使用静态成员

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);

}

  • 由于用 静态成员sBackground 缓存了drawable对象,所以activity加载速度会加快,但是这样做是错误的。因为在android 2.3系统上,它会导致activity销毁后无法被系统回收。

label .setBackgroundDrawable()调用会将label赋值给sBackground的成员变量 mCallback

上面代码意味着:sBackground(GC Root)会持有TextView对象,而TextView持有Activity对象。所以导致Activity对象无法被系统回收。

下面看看android4.0为了避免上述问题所做的改进。

  • 先看看android 2.3的Drawable.Java对setCallback的实现:

public final void setCallback(Callback cb){

mCallback = cb;

}

// 在android 2.3中要避免内存泄漏也是可以做到的,

// 在activity的onDestroy时调用

// sBackgroundDrawable.setCallback(null)。

  • 再看看android 4.0的Drawable.Java对setCallback的实现:

public final void setCallback(Callback cb){

mCallback = newWeakReference (cb);

}

以上2个例子的内存泄漏都是因为 Activity的 引用的生命周期 超越了Activity 对象的生命周期。也就是常说的 Context泄漏,因为activity就是context。

3. 避免context相关的内存泄漏,需要注意以下几点
  • 不要对activity的context长期引用

( 一个activity的引用的生存周期应该和activity的生命周期相同 )

  • 如果可以的话,尽量使用关于application的context来替代和activity相关的context

  • 如果一个acitivity的非静态内部类的生命周期不受控制,那么避免使用它;正确的方法是 使用一个静态的内部类,并且对它的外部类有一WeakReference,就像在ViewRootImpl中内部类W所做的那样

4. 使用handler时的内存问题
  1. 我们知道,Handler通过发送Message与主线程交互。
  • Message发出之后是存储在MessageQueue中的,有些Message也不是马上就被处理的。

  • 在Message中存在一个 target,是Handler的一个引用,如果Message在Queue中存在的时间越长,就会导致Handler无法被回收。

  • 如果Handler是非静态的,则会导致Activity或者Service不会被回收。 所以正确处理Handler等之类的内部类,应该将自己的Handler定义为静态内部类

  1. HandlerThread的使用也需要注意:
  • 当我们在activity里面创建了一个HandlerThread,代码如下:

public classMainActivity extends Activity

{

@Override

public void onCreate(BundlesavedInstanceState)

{

super.onCreate(savedInstanceState);

setContentView(R.layout.activity_main);

Thread mThread = newHandlerThread(“demo”, Process.THREAD_PRIORITY_BACKGROUND);

mThread.start();

MyHandler mHandler = new MyHandler( mThread.getLooper( ) );

}

@Override

public void onDestroy()

{

super.onDestroy();

// mThread.getLooper().quit();

}

}

  • 这个代码存在泄漏问题,因为 HandlerThread的run方法是一个死循环,它不会自己结束,线程的生命周期超过了activity生命周期,当横竖屏切换,HandlerThread线程的数量会随着activity重建次数的增加而增加。

  • 应该在onDestroy时将线程停止掉:mThread.getLooper().quit();

另外,对于不是HandlerThread的线程,也应该确保activity消耗后,线程已经终止,可以这样做:在onDestroy时调用 mThread.join();

join( ) 的作用是:“等待该线程终止”,这里需要理解的就是该线程是指的主线程等待子线程的终止。也就是:在子线程调用了join()方法后面的代码,只有等到子线程结束了才能执行。

5. 注册某个对象后未反注册

比如 注册广播接收器注册观察者 等等。

  • 假设我们希望在锁屏界面(LockScreen)中,监听系统中的电话服务以获取一些信息(如信号强度等),则可以在LockScreen中定义一个PhoneStateListener的对象,同时将它 注册TelephonyManager服务中。对于LockScreen对象,当需要显示锁屏界面的时候就会创建一个LockScreen对象,而当锁屏界面消失的时候LockScreen对象就会被释放掉。

  • 但是如果 在释放LockScreen对象的时候忘记取消我们之前注册的PhoneStateListener对象,则会导致LockScreen无法被GC回收。如果不断的使锁屏界面显示和消失,则最终会由于大量的LockScreen对象没有办法被回收而引起OutOfMemory,使得system_process进程挂掉。

虽然有些系统程序,它本身好像是可以自动取消注册的(当然不及时),但是我们还是 应该在我们的程序中明确的取消注册,程序结束时应该把所有的注册都取消掉。

6. 集合中对象没清理造成的内存泄露

我们通常把一些对象的引用加入到了集合中,当我们不需要该对象时,如果没有把它的引用从集合中清理掉,这样这个集合就会越来越大。如果这个集合是static的话,那情况就更严重了。

  • 比如某公司的ROM的锁屏曾经就存在内存泄漏问题:

  • 这个泄漏是因为LockScreen**每次显示时会注册几个callback**,它们保存在

KeyguardUpdateMonitor的ArrayList<InfoCallback>

ArrayList<SimStateCallback>

等ArrayList实例中。但是在LockScreen**解锁后,这些callback没有被remove掉**,导致ArrayList不断增大, callback对象不断增多。这些callback对象的size并不大,heap增长比较缓慢,需要长时间地使用手机才能出现OOM,由于锁屏是驻留在system_server进程里,所以导致结果是手机重启。

7. 资源对象没关闭造成的内存泄露
  • 资源性对象 比如(CursorFile文件等) 往往都用了一些缓冲,我们在不使用的时候,应该及时关闭它们,以便它们的缓冲及时回收内存。它们的缓冲不仅存在于Java虚拟机内,还存在于Java虚拟机外。

  • 如果我们仅仅是把它的引用设置为null,而不关闭它们,往往会造成内存泄露。因为有些资源性对象,比如SQLiteCursor(在析构函数finalize(),如果我们没有关闭它,它自己会调close()关闭),如果我们没有关闭它,系统在回收它时也会关闭它,但是这样的效率太低了。因此对于资源性对象在不使用的时候,应该立即调用它的close()函数,将其关闭掉,然后再置为null.

  • 在我们的程序退出时一定要确保我们的资源性对象已经关闭

8. 一些不良代码成内存压力

有些代码并不造成内存泄露,但是它们或是 对不使用的内存没进行有效及时的释放,或是没有有效的利用已有的对象而是频繁的申请新内存,对内存的回收和分配造成很大影响的。

1) Bitmap使用不当
  • 及时的销毁

在用完Bitmap时,要及时的bitmap.recycle( )掉。

注意,recycle( )并不能确定立即就会将Bitmap释放掉,但是会给虚拟机一个暗示:“该图片可以释放了”。

  • 设置采样率

有时候,我们要显示的区域很小,没有必要将整个图片都加载出来,而只需要记载一个缩小过的图片,这时候可以设置一定的采样率,那么就可以大大减小占用的内存。如下面的代码:

private ImageView preview;

BitmapFactory.Options options = newBitmapFactory.Options();

// 图片宽高都为原来的二分之一,即图片为原来的四分之一

options.inSampleSize = 2;

Bitmap bitmap = BitmapFactory.decodeStream(cr.openInputStream(uri),

null, options); preview.setImageBitmap(bitmap);

  • 巧妙的运用软引用(SoftRefrence)

有些时候,我们使用Bitmap后没有保留对它的引用,因此就无法调用Recycle函数。这时候巧妙的运用软引用,可以使Bitmap在内存快不足时得到有效的释放。如下:

SoftReference bitmap_ref = new SoftReference(

BitmapFactory.decodeStream(inputstream));

if (bitmap_ref .get() != null) {

bitmap_ref.get().recycle();

}

2) 构造Adapter时,没有使用缓存的 convertView
  • 初始时ListView会从BaseAdapter中根据当前的屏幕布局实例化一定数量的view对象,同时ListView会将这些view对象缓存起来。

  • 当向上滚动ListView时,原先位于最上面的list item的view对象会被回收,然后被用来构造新出现的最下面的list item。这个构造过程就是由getView()方法完成的,getView()的第二个形参 View convertView就是被缓存起来的list item的view对象 ( 初始化时缓存中没有 view 对象,则 convertView 是 null )。

由此可以看出,如果我们不去使用convertView,而是每次都在getView()中重新实例化一个View对象的话,即浪费时间,也造成内存垃圾,给垃圾回收增加压力,如果垃圾回收来不及的话,虚拟机将不得不给该应用进程分配更多的内存,造成不必要的内存开支。

3) 不要在经常调用的方法中创建对象,尤其是忌讳在循环中创建对象。

可以适当的使用 hashtablevector 创建一组对象容器,然后从容器中去取那些对象,而不用每次 new 之后又丢弃。

9. 查询数据库而没有关闭Cursor

在Android中,Cursor是很常用的一个对象,但在写代码时,经常会有人忘记调用close, 或者因为代码逻辑问题状况导致close未被调用

  • 通常,在Activity中,我们可以调用startManagingCursor或直接使用managedQuery让Activity自动管理Cursor对象。

但需要注意的是,当Activity结束后,Cursor将不再可用!

  • 若操作Cursor的代码和UI不同步(如后台线程),需要先判断Activity是否已经结束,或者在调用OnDestroy前,先等待后台线程结束。

  • 除此之外,以下也是比较常见的Cursor不会被关闭的情况:

自我介绍一下,小编13年上海交大毕业,曾经在小公司待过,也去过华为、OPPO等大厂,18年进入阿里一直到现在。

深知大多数初中级Android工程师,想要提升技能,往往是自己摸索成长或者是报班学习,但对于培训机构动则近万的学费,着实压力不小。自己不成体系的自学效果低效又漫长,而且极易碰到天花板技术停滞不前!

因此收集整理了一份《2024年Android移动开发全套学习资料》,初衷也很简单,就是希望能够帮助到想自学提升又不知道该从何学起的朋友,同时减轻大家的负担。

img

img

img

img

既有适合小白学习的零基础资料,也有适合3年以上经验的小伙伴深入学习提升的进阶课程,基本涵盖了95%以上Android开发知识点,真正体系化!

由于文件比较大,这里只是将部分目录截图出来,每个节点里面都包含大厂面经、学习笔记、源码讲义、实战项目、讲解视频,并且会持续更新!

如果你觉得这些内容对你有帮助,可以扫码获取!!(备注:Android)

文末

好了,今天的分享就到这里,如果你对在面试中遇到的问题,或者刚毕业及工作几年迷茫不知道该如何准备面试并突破现状提升自己,对于自己的未来还不够了解不知道给如何规划,可以来看看同行们都是如何突破现状,怎么学习的,来吸收他们的面试以及工作经验完善自己的之后的面试计划及职业规划。

这里放上一部分我工作以来以及参与过的大大小小的面试收集总结出来的相关的几十套腾讯、头条、阿里、美团等公司21年的面试专题,其中把技术点整理成了视频和PDF(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分免费分享给大家,主要还是希望大家在如今大环境不好的情况下面试能够顺利一点,希望可以帮助到大家~

还有 高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

【Android核心高级技术PDF文档,BAT大厂面试真题解析】

【延伸Android必备知识点】

这里只是整理出来的部分面试题,后续会持续更新,希望通过这些高级面试题能够降低面试Android岗位的门槛,让更多的Android工程师理解Android系统,掌握Android系统。喜欢的话麻烦点击一个喜欢在关注一下~

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

(实际上比预期多花了不少精力),包含知识脉络 + 诸多细节,由于篇幅有限,这里以图片的形式给大家展示一部分免费分享给大家,主要还是希望大家在如今大环境不好的情况下面试能够顺利一点,希望可以帮助到大家~

还有 高级架构技术进阶脑图、Android开发面试专题资料,高级进阶架构资料 帮助大家学习提升进阶,也节省大家在网上搜索资料的时间来学习,也可以分享给身边好友一起学习。

【Android核心高级技术PDF文档,BAT大厂面试真题解析】

[外链图片转存中…(img-xZQHsUab-1713087088080)]

【延伸Android必备知识点】

[外链图片转存中…(img-JKNNluOl-1713087088080)]

这里只是整理出来的部分面试题,后续会持续更新,希望通过这些高级面试题能够降低面试Android岗位的门槛,让更多的Android工程师理解Android系统,掌握Android系统。喜欢的话麻烦点击一个喜欢在关注一下~

《互联网大厂面试真题解析、进阶开发核心学习笔记、全套讲解视频、实战项目源码讲义》点击传送门即可获取!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值