Android App质量提升手册

本文从内存泄漏、溢出、抖动,界面卡顿,过度绘制,启动分析层面给出的质量改进思路,结合严苛模式,可帮助开发人员快速查错以找到解决方案。

一、 内存泄漏

内存泄漏是指由于代码编写不当导致不再使用的对象无法得到及时释放。内存泄漏产生的内存垃圾不仅浪费资源,拖慢运行效率,甚至还可能造成内存溢出,直接导致应用崩溃。

对于Android应用,比较容易发生泄漏的是Activity、Fragment对象,此类对象的共性是其都有一定的生命周期。以Activity为例,一个Activity实例的生命起始于onCreate(),终结于onDestroy()。当一个Activity不再使用时,系统会调用回调方法Activity.onDestroy()方法做一些清理操作。但是对于Activity对象本身所占内存,则完全由虚拟机的垃圾回收器来完成回收。垃圾回收器会检查该实例是否被持有强引用,如果存在指向该对象的强引用,则不会回收其所占内存空间,这块内存空间也就成了内存垃圾。由此可见内存泄漏是由不当的强引用导致的。

1.1 对象的引用链

从GC ROOT到泄漏对象的引用链能精准地定位导致内存泄漏的原因。对象无法被垃圾回收器回收,一定是由于GC ROOT直接或间接持有了它的强引用。

常见的GCROOT有:声明为static的变量,未停止的线程,Application对象,甚至是栈内存中的局部变量。

1.2 Android中常见的内存泄露

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

编程过程中,我们常常会把一些对象加入到集合中。在我们不再需要该对象时,如果没有及时把它从集合中清理掉,就会导致这个集合占用的内存越来越大。同时如果这个集合是静态的话,那情况就更严重了。如下的代码段中在每次启动Activity的时候都往静态集合中添加了一个对象,如果Activity被频繁启动,set将不断变大,影响APP的正常运行。

所以,集合中不再使用的对象应及时释放掉。上述代码应该在Activity的onDestroy()方法中,及时清理set里的元素,避免无用对象继续存在强引用,例如:

这样可以保证set持有的强引用都被释放。      

b. 单例模式造成的内存泄漏

单例的静态特性使得其生命周期可能跟应用的生命周期一样长,如果使用不恰当的话,很容易造成内存泄漏。

如下代码是一个简单的单例模式实现:

在创建单例的时候,如果我们传入当前Activity的Context,例如:

单例testContextHelper里面一直保存着该Activity的引用,当这个Context 对应的 Activity 退出时,由于该 Context 的引用一直被单例对象持有,所以该Activity占用的内存并不会被回收,造成泄漏。在使用单例模式时,一定要避免持有短生命周期对象的引用,比如上述代码在引用Context时可以使用Application的Context代替Activity的Context,即:

 

因为Application在应用的运行过程中一直存在,不会退出。 

c. 非静态内部类创建静态实例造成的内存泄漏

在启动频繁的Activity中,为了避免反复创建某些资源,提高加载速度,我们可能会在Activity内部创建一个静态实例,每次启动Activity时都会使用该实例,如下代码:

 

此时Activity内部有一个静态单例,且为非静态内部类的实例。由于非静态内部类默认会持有外部类的引用,并且该类创建了一个静态实例,该实例的生命周期和应用的一样长,这就导致了该静态实例一直会持有该Activity的引用,导致Activity的内存资源不能正常回收。为了避免这一问题,在使用过程中,正确的做法是将内部类设为静态类或者变成单独的类。 

d. 使用handler时的内存问题

在Android应用中,Handler通过发送Message与其他线程交互,发出的Message被存储在目标线程的MessageQueue中的,并且Message不一定马上就被处理,驻留时间可能比较久。比如我们用Handler发送一个延时比较久的Message:

而Message中持有Handler实例的强引用,如果Message在Queue中一直存在,就会导致Handler实例无法被回收,而Handler持有Activity的强引用,Activity对象也不会被回收,这就造成了实例泄露。所以,在创建Handler时,最好使用弱引用来引用目标Activity对象,比如:

这样可以避免由于Handler持有强引用导致Activity无法回收。 

e. 静态成员变量造成的内存泄露

如果成员变量被声明为 static,其生命周期将与整个应用进程的生命周期一样。如果静态变量直接或间接强引用了某一短生命周期对象(比如Activity),这会导致即使app切到后台,这部分内存也不会被释放。下面的错误示范代码中,在Activity启动的时候,直接将其引用赋给了静态变量obj,会导致该Activity一直不能被回收,导致内存泄露。

 

因此,在使用静态变量时,应该避免其持有短生命周期对象的强引用,可以使用弱引用来代替强引用。 

f. 资源未关闭造成的内存泄漏

对于使用了BroadcastReceiver,ContentObserver,File,游标Cursor,Stream,Bitmap等资源的使用,应该在Activity销毁时及时关闭或者注销,否则这些资源将可能不会被回收,造成内存泄漏。虽然有些系统程序,它本身可以自动取消注册的(非即时),但是我们还是应该在我们的程序中明确的取消注册,程序结束时应该把所有的注册都取消掉。

二、 内存溢出

内存溢出(OOM, Out OfMemory)是指当已存在的对象的占用了绝大部分或者全部分配给该进程的内存空间时,如果进程再申请新的内存空间,由于没有空余内存可用于分配,或可分配的内存不够满足申请者的需求,此时系统就会抛出内存溢出异常。

常见的内存溢出原因

很大一部分内存溢出都是由于内存泄露导致,由于已分配的内存被泄露对象占用并且无法释放,随着泄露的对象实例越来越多,导致可用内存越来越少,最终当内存耗尽时,系统就会抛出内存溢出异常。此时只要解决了内存泄露,也就解决了内存溢出。

另一个内存溢出的重要原因就是应用加载了多个占用内存较多的对象。比如应用在运行过程中加载并保存了多个较大的Bitmap,导致可用内存急剧减少。因此,在代码编写过程中,对于可能占据大量内存空间的对象,我们应该使用软引用或虚引用持有该对象,使得系统GC能在内存吃紧时回收该对象释放空间。并且在不使用Bitmap时,应及时recycle,主动释放内存空间。

三、内存抖动

内存抖动指的是短时间内大量对象被创建和回收。由于短时间内产生了大量的对象,需要分配大量内存,此时需要垃圾回收器(GC)频繁工作,回收不再使用的对象来腾出内存空间。GC的频繁启动占用了一定的系统资源,最终影响应用表现。

常见的内存抖动

常见的内存抖动主要是由于在循环或其他场合中不停地创建新对象,并且短时间内这些对象又被释放。瞬间产生大量的对象会严重占用内存区域,当达到阀值,剩余空间不够的时候触发GC。即使每次分配的对象占用了很少的内存,但是他们叠加在一起会增加Heap的压力。GC启动时会占用CPU等资源,直接导致应用运行受到影响,可能出现界面操作不流畅等现象。

四、界面卡顿

界面卡顿指的是短时间内界面对用户操作没有响应。应用在出现卡顿的时候,就算知道是哪个页面出了问题,但是很难定位到具体的代码。应用卡顿检测就是帮助您快速定位卡顿的具体位置,方便您进行针对性的修复。 

常见的界面卡顿原因

Android应用的UI绘制和用户操作消息分发都发生在应用主线程,如果主线程来不及处理UI更新和响应用户操作,用户就会感觉应用发生了卡顿。因此卡顿发生时尝尝伴随着主线程阻塞。如果在主线程中进行磁盘读写、网络操作或者大量计算时,尝尝会导致主线程被阻塞,发生界面卡顿。

五、过度绘制

过度绘制一般指的是屏幕上的某些区域在一帧中被多次绘制,一般是在界面的同一个地方叠加了多个控件。这样会加重GPU的工作负担,可能导致应用运行过程中频繁掉帧,影响用户体验。

详细介绍

当手机开启过度绘制时,屏幕上会标记发生过度绘制的区域,并根据不同的绘制次数使用不同的颜色,颜色标识从好到差依次是:蓝色-绿色-淡红色-红色,分别代表该区域被绘制1次、2次、3次和4次。一般情况下,最好把绘制控制在2次以下,3次绘制有时候是不能避免的,尽量避免,4次的绘制基本上是不允许的。

为了减少过度绘制,开发者应减少复杂的、层级较多的布局,去掉多余的背景色。简单的界面尽量使用线性布局;较为复杂的界面可以使用相对布局,避免嵌套过多的线性布局。可以使用ViewStub来动态加载界面。

六、启动分析

启动分析通过分析应用启动过程产生的trace文件来得到应用的启动时间等信息。通常来说,Android应用的启动方式分为两种:冷启动和热启动。

冷启动:当启动应用时,后台没有该应用的进程,此时系统会创建一个新的进程分配给该应用。冷启动因为系统会创建一个新的进程分配给它,所以会先创建和初始化Application类,随后创建和初始化MainActivity类(包括一系列的测量、布局、绘制),最后显示在界面上。

热启动:当启动应用时,后台已有该应用的进程(例:按back键、home键,应用虽然会退出,但是该应用的进程是依然会保留在后台,可进入任务列表查看),这种启动会从已有的进程中来启动应用。热启动因为会从已有的进程中来启动,所以热启动就不会创建新的Application,而是直接创建和初始化MainActivity,而不必创建和初始化Application,因为一个应用从新进程的创建到进程的销毁,Application只会初始化一次。一般来讲,热启动时间都会在一定程度上小于冷启动时间。

七、严苛模式(StrictMode)

严苛模式(StrictMode)是一个开发辅助工具,可以帮助开发者发现那些由于编码过程中不注意而造成的问题。 

详细介绍

StrictMode经常用于捕获那些在应用主线程中进行的磁盘读写操作和网络请求。由于应用主线程是接收UI操作消息和执行界面渲染的地方,为了使应用运行更加流畅和更快响应,请尽量不要在主线程执行磁盘操作和网络请求。当然,这也是避免系统弹出ANR对话框和提高应用稳定性的好方法。一旦检测到违反策略(policyviolation),系统将会输出一条相关的日志,其一般包含一个调用栈,来显示应用在何处发生违例。

注意:尽管Android设备的磁盘一般都是闪存盘,然而实际中很多设备只能以很有限的并发数来操作文件系统。虽然磁盘读写很快,但是具体过程中可能由于其他进程占用了I/O接口,等待的过程会导致整个磁盘操作流程比较慢。如果可以,请尽量假设磁盘读写是一个比较耗时的操作。

StricMode除了可以检测主线程的磁盘操作和网络请求以外,还可以发现主线程中执行时间较长的方法。当应用中有继承了Closeable接口的对象没有关闭的时候,例如文件流等,或者没有使用HTTPS进行网络请求,或者同一个Activity的实例太多,StrictMode都会给出提示。其能发现的错误主要包括:

a.应用在主线程中进行磁盘读写;

b.应用在主线程中进行网络请求;

c.应用在主线程中的某些自定义方法的执行时间比较长;

d.SQLCursor对象在使用之后没有关闭;

e.继承了Closeable接口的对象在使用之后没有关闭;

f.某一Activity有较多的实例;

g.文件读取接口暴露给外部应用;

h.注册某些对象(广播接收器、观察者、Listener等)后没有取消注册;

i.没有使用加密网络(HTTPS)进行网络数据传输。

Written on March 13, 2017

请我喝杯咖啡吧