转载注明出处:http://blog.csdn.net/xiaohanluo/article/details/76546209
Android系统每隔16ms发出VSYNC信号,对UI进行渲染,如果每次渲染都成功,就能够达到流畅的画面所需要的60fps,为了能够实现60fps,这意味着程序的大多数操作都必须在16ms内完成,时间超出16ms越多,丢的帧就越多。
假设我们更新屏幕的背景图片,需要24ms来做这次运算。当系统在第一个16ms时刷新界面,然而我们的运算还没有结束,无法绘出图片。当系统隔16ms再发一次VSYNC信息重绘界面时,用户才会看到更新后的图片。也就是说用户是32ms后看到了这次刷新(注意,并不是24ms),这就是丢帧。
大多数多用感知到卡顿等问题最主要的根源是渲染问题,而导致渲染问题的原因是性能问题,为了保证程序正常的使用,性能方面需要着重注意,本篇针对的性能优化是从一些平时常见的细节入手,
丢帧只是用户能感知到的表面现象,严重的会引起程序卡顿甚至ANR,深层次的原因是代码中有比较耗时的操作阻塞到了主线程,也就是性能问题。
过度绘制(Overdraw)是指屏幕上同一个像素点在同一帧时间内被绘制多次。在层级复杂的UI结构中,如果不可见的UI也被绘制,会导致某些像素区被绘制多次,浪费大量的CPU以及GPU资源。
在手机的开发者选项
中,打开显示布局边界
即可查看页面的绘制信息。
有四中颜色,蓝色,淡绿,淡红,深红分别代表着不同的绘制信息。蓝色代表一次绘制,淡绿代表两次绘制,淡红代表三次绘制,深红代表四及以上次绘制。
我们的目的是尽量减少红色的绘制信息,方法有两种。
上面提到简化布局可以减少过度绘制的问题,布局优化可以从下面几方面入手。
LinearLayout
,因为RelativeLayout
在measure
会比LinearLayout
多出一次, LinearLayout
使用了weight
后,性能依然会比RelativeLayout
好RelativeLayout
的measure
过程中,如果出现子View和布局本身高度不同时候,还会触发measure
过程,解决方法很简单,使用padding
代替marigin
关于这块的具体详情,参考下面链接:
带你从源代码详细分析View的绘制过程
Android布局优化之ViewStub、include、merge使用与源码分析
Android中RelativeLayout和LinearLayout性能分析
这里I/O操作仅针对SharedPreferences
,因为这个基本是在Android中使用最多的储存库了。
读操作一般不会阻塞到主线程,如果读取的数据比较大而且需要有大量的处理操作,直接开子线程,在子线程处理。
写操作比读操作复杂一些,向SharedPreferences
写入数据时候,有commit
和apply
两个方法。
commit和apply方法区别在于同步写入和异步写入,以及是否需要返回值。在不需要返回值的情况下,使用apply方法可以极大的提高性能。
在使用lint静态扫描代码时候,会建议使用apply
去替代commit
,说明官方也是支持使用apply
方法的。
但是!!!
并不是是说apply不会阻塞到主线程,它也有可能会阻塞到主线程。详情请看这篇文章SharedPreferences调用导致的ANR分析。
长话短说。
使用apply方法时候,会向QueuedWork
队列中添加一个等待写入操作完成的线程,只有当写入操作完成后,才会从QueuedWork
将等待线程线程移除掉。
而主线程中,Service的启动和stop
以及Activity的onPause()
和onStop()
生命周期都会等待其他异步线程完成,才会继续执行。
例如停止Service的时候代码如下(ActivityThread类中):
private void handleStopService(IBinder token) {
···
QueuedWork.waitToFinish(); // 等待其他异步线程完成
···
}
可以看到,如果使用apply方法写入大量复杂数据,确实有可能会阻塞到主线程,甚至可能导致ANR。
优化方法:
大部分情况下,与后台交互使用的数据是gson格式。相信很多是直接使用google官方的Gson类来序列化和反序列化的。
我在项目中就遇见了有使用Gson反序列化复杂数据时候,造成的卡顿现象。
Gson序列化和反序列化是可以优化的。
// 反序列化
public List readJsonStream(InputStream in) throws IOException {
JsonReader reader = new JsonReader(new InputStreamReader(in, "UTF-8"));
List messages = new ArrayList();
reader.beginArray();
while (reader.hasNext()) {
Message message = gson.fromJson(reader, Message.class);
messages.add(message);
}
reader.endArray();
reader.close();
return messages;
}
// 序列化
public void writeJsonStream(OutputStream out, List messages) throws IOException {
JsonWriter writer = new JsonWriter(new OutputStreamWriter(out, "UTF-8"));
writer.setIndent(" ");
writer.beginArray();
for (Message message : messages) {
gson.toJson(message, Message.class, writer);
}
writer.endArray();
writer.close();
}
大部分框架或者SDK为了解耦,或多或少的会使用到反射,反射本身会性能就会比直接调用差很多。
有时候会大批量使用反射去实例化对象,所以不要在那些会被反射实例化的类的构造函数中做太多的事情。
建议在获取实例化对象后,再使用对象直接调用初始化方法。
在代码中有时候可能会发生异常,所以一般使用try/catch来做保护,避免程序崩溃。
一旦有异常发生,系统会耗费资源去处理,本身对内存和CPU就会有消耗。
所以不能因为有了try/catch而不顾代码质量,要尽量保证没有大量的异常出现,我在项目中见过大量空指针异常出现,这种异常我们可以在代码层面就直接避免。
频繁的GC也会对性能造成影响,严重的会导致卡顿或者ANR。
频繁GC原因有两个
在项目中避免短时间内突然创建大量的对象。
获取设备&应用基本信息,比如说包名、IMEI信息等等,是比较耗时的操作,可以进行一些优化。
目前自己遇见并优化过的都在以上列出来了,后续会慢慢积累。
关于性能优化方面知识,官方也出过视频Android Performance Patterns,网上也有很多翻译的文章,写得都比较详细,这篇文章是自身从平时一些常见点入手做的一些总结,希望对大家有帮助。