android 内存优化 性能优化,【Android面试周】性能优化之内存篇

性能优化一直是高级Android工程师必问的,这也往往涉及到对Java、JVM、Android运行机制、监控工具使用等全方位的知识点,是很好的面试方向。

本篇会围绕Android与Java的性能优化专项之内存展开并深入。

内存是Android运行性能至关重要的一项指标,每个进程能使用的内存是有限的。不合理的使用内存会导致频繁的GC、甚至发生OOM,过多GC会导致App卡顿,而内存泄漏或者内存抖动都可以导致OOM,这是无法接受的。

因此,对于一个合格的高级Android工程师,必须保持对内存的高度敏感性,本文会针对内存提出一系列性能优化手段。

什么时候会导致频繁GC

内存抖动

短时间内创建了大量对象同时又被快速释放。比如在一个大循环里去不断创建对象,会导致频繁gc;

内存泄漏

内存泄漏会导致可用内存逐渐变少,而且内存碎片加多,这也会增多gc次数,甚至可能发生OOM

一次申请太大内存空间

由于内存碎片的存在,就算内存本身足够,但由于碎片导致无法找到一块大空间,这也会触发gc;

内存优化准则

1. 能不创建的对象就不创建

比如字符串拼接,可以手动使用StringBuilder,而不是使用"+","+"被编译器优化后会每次创建StringBuilder对象,造成浪费;

而且,尤其注意在主线程里不要过多创建对象。因为在GC时会锁住堆内存,此时请求分配的线程也会被挂起,这显然会导致主线程的卡顿。所以在一些主线程高频函数,如onDraw,onTouchEvent里不要去创建对象。

2. 尽可能复用已经创建的对象

还是StringBuilder的例子,基于一个StringBuilder可以通过SetLength(0)支持很多次的字符串拼接。

多使用系统提供的对象池,比如线程池,Long、Integer、Short等包类型里的缓存值(通过valueOf取),列表view的复用等

3. 防止内存泄漏

内存一旦发生泄漏,意味着堆里有一块区域持续被不再使用的变量占据,这自然会导致可用内存减少而发生gc,甚至OOM。

一般内存泄漏有几下几种情况:

I. 长生命周期持有短生命周期引用,如Activity泄漏

一般是长生命周期的变量持有Activity引用,导致在gc时无法标记Activity从而无法回收,而Activity本身一般会引用到非常多的资源如View, Image,则这些大块资源均无法回收。

这种时候,我们可以用WeakReference来弱引用;针对Activity这种场景可以使用LeakCanary来进行监控,它会在Activity onDestroy后监控其引用是否被释放,若未释放则主动触发一次gc,gc后如果仍未释放,则会通知开发者。

II. 静态变量和单例滥用

静态变量一般是在装载类里的链接时进行内存分配,初始化时进行赋值。关键是,静态变量会伴随Android进程整个生命周期,如果引用了某块堆内存,则该内存无法被回收。单例也是类似。我们在开发中不应该依赖太多静态变量和单例。

III. finalize来兜底

对于一些较大内存的对象,可以考虑利用finalize方法,防止在忘记释放时主动进行一些清理工作。像SQLiteDatabase、FileInputStream等都会在finalize方法里面进行一些清理工作,如关闭数据库。

不过,finalize方法并不是万能的,它也会导致其他问题,后续再说。

IV. 合理应用Java各种引用

Java里提供了四种引用:

强引用StrongReference:这种引用的对象不会被GC回收,在堆里存活时间最久;

弱引用WeakReference:在GC时一旦遇到WeakReference,无论堆内存空间是否足够都会进行回收,不过考虑到GC线程优先级很低,所以WeakReference能存活较长时间;(附:当WeakReference对象被回收后,它自身可被放入一个ReferenceQueue里,LeakCanary里便应用了这个特性来检测Activity是否已被回收)

软引用SoftReference:当堆内存不足时,会对SoftReference进行回收,它存活的时间一般长于WeakReference。(附:SoftReference是在第三次内存分配失败时进行回收。流程:第一次分配失败->GC->第二次分配失败->扩容->第三次仍分配失败->清理SoftReference)

虚引用PhantomReference:等于没有引用,随时随地都会被回收

Dalvik与Art的垃圾回收区别

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值