上一期分享了android内存优化的一些总结,这一期说说我认为的好的编码习惯,然后下一期会做安卓数据库优化的一些总结,逐渐的会将一些性能优化点总结分享出来,肯定是不够全面的希望不足的地方欢迎指出。
良好的编码习惯除开编码规范这个不说外,还有很多影响内存,流畅度,耗电量的地方都是需要注意的。我会随时补充进来我遇到的相关的好的编码习惯,但不太容易做到一下就全面。
1、内存控制
1.1 已知数量时,对相应的数据结构赋值相应的大小
new ArrayList(5);
1.2 勿在循环调用的地方new对象
比如在adapter中的getView去new OnClickListener,View的onDraw方法new Paint,这些循环多次调用的地方会导致多个对象的产生,不进造成内存增加也会浪费时间。
1.3 匿名内部类的使用
上一篇博文提到了原因,匿名内部类会持有外部引用,所以这里改成静态内部类的,同时使用弱引用的方式。
1.4 enum的使用
2、节省时间
2.1 用变量替代计算
不论是for循环中,还是自己实现一些代码通过用一个变量记录值,比循环计算得出来的值更加节省时间。比如
for(int i =0;i < array.length;i++) 可以替换成
final int size = array.length;
for(int i = 0; i< size;i++),这样可以减少循环过程中的计算
不过据说现在foreach的方式是最快的,有兴趣的朋友可以测试下。
2.2 字符串的拼接使用StringBuilder或者StringBuffer
比如"select * from " + Tables.table_name+" where id = "+sid+" limit 1";字符串的拼接不光是时间损耗,内存也会损耗。如果一个字符串后面跟着一个+号,再后面是任何的类型表达式:
string_exp + any_exp
Java编译器会把它变成:
new StringBuilder().append( string_exp ).append( any_exp ).toString()
如果表达式里有多个+号的话,后面相应也会多多几个StringBuilder.append的调用,最后才是toString方法。
而StringBuilder(String)这个构造方法会分配一块16个字符的内存缓冲区。因此,如果后面拼接的字符不超过16的话,StringBuilder不需要再重新分配内存,不过如果超过16个字符的话StringBuilder会扩充自己的缓冲区。最后调用toString方法的时候,会拷贝StringBuilder里面的缓冲区,新生成一个String对象返回。
关于字符串拼接的优化除了StringBuilder或者buffer外还可以做的更好一点,这也是一次面试得到的经验,可以下一次分享给大家。
2.3 注意强制类型转换
使用Integer.parseInt()而不是(int)Integer.valueOf()然后去做对比,减少类型强制转换的时间
2.4 对常量使用Static Final修饰符,
2.5 避免内部getters和setters(有争议)
性能上直接访问变量更加有效率,但是软件工程上就对外暴露了同时也不容易面向扩展了,所以这里只是提出来,并不一定要遵守。
2.6 在私有内部类中,考虑用包访问权限替代私有访问权限
访问外部内的私有方法,会让虚拟机生成黏合方法,也就是getters和setters的方法从而降低了读取的性能。
2.7 线程的优先级设定
首先不要同时使用过多的线程,一来难以管理,二来也会耗费cpu资源,导致影响UI线程,所以在使用线程时,在执行优先级不高的情况下,指定线程最低优先级或者后台优先级等,保证UI线程的优先级,比如:
new Thread(new Runnable() {
@Override
public void run() {
Process.setThreadPriority(Process.THREAD_PRIORITY_LOWEST);
}
}).start();
2.8 使用System.arraycopy ()代替通过来循环复制数组
2.9 避免循环内部有过多依赖和跳转,使cpu能流水起来
比如:
for(int i =0; i < N; i++){
if(i <100)
a[i] +=5;
else if(i <200)
a[i] +=10;
else
a[i] +=20;
}
改成三个循环
for(int i = 0; i < 100; i++)
{
a[i] += 5;
}
for(int i = 100; i < 200; i++)
{
a[i] += 10;
}
for(int i = 200; i < N; i++)
{
a[i] += 20;
}
3、对应的生命周期
3.1 set(listener)对于set(null)
3.2 add对应remove
3.3 open对应close
4、屏幕网络CPU耗电量杀手
4.1 选择合适的wakelock模式
PARTIAL_WAKE_LOCK:保持CPU运转,屏幕和键盘灯有可能是关闭的。
SCREEN_DIM_WAKE_LOCK:保持CPU运转,允许保持屏幕显示但有可能是灰的,允许关闭键盘灯
SCREEN_BRIGHT_WAKE_LOCK:保持CPU运转,保持屏幕高亮显示,允许关闭键盘灯
FULL_WAKE_LOCK:保持CPU运转,保持屏幕高亮显示,键盘灯也保持亮度
ACQUIRE_CAUSES_WAKEUP:不会唤醒设备,强制屏幕马上高亮显示,键盘灯开启。有一个例外,如果有notification弹出的话,会唤醒设备。
ON_AFTER_RELEASE:WakeLock被释放后,维持屏幕亮度一小段时间,减少WakeLock循环时的闪烁情况
用完必须要记得释放。
4.2 选择合适的alarmmanage模式
AlarmManager会维持一个cpu的wakelock。这样能保证电话休眠时,也能处理alarm的广播,有4种Alarm类型:
1)RTC_WAKEUP
在指定的时刻(设置Alarm的时候),唤醒设备来触发Intent。
2)RTC
在一个显式的时间触发Intent,但不唤醒设备。
3)ELAPSED_REALTIME
从设备启动后,如果流逝的时间达到总时间,那么触发Intent,但不唤醒设备。流逝的时间包括设备睡眠的任何时间。注意一点的是,时间流逝的计算点是自从它最后一次启动算起。
4)ELAPSED_REALTIME_WAKEUP
从设备启动后,达到流逝的总时间后,如果需要将唤醒设备并触发Intent。
4.3 慎用轮询
这玩意耗电很大,当然很多时候可能是配合上面的AlarmManager使用,可以在不同的界面或者功能,定义不同的轮询时间来做优化等。
5、IO优化
5.1 使用缓存区
通过缓存减少磁盘读取次数
5.2 缓冲块的IO要比缓冲流IO要快
6、功能范围
6.1 合理的synchronized范围
6.2 合理的try/catch范围
try/catch住爆出异常的区域,而不是一个大的函数。
网友评论