App性能优化学习
App性能优化
对于一个Android开发,一个好的App:
- 流畅
- 稳定
- 省电省流量
- 安装包小
流畅
使用时避免出现卡顿,提高响应速度
卡顿
根本原因:
- 界面绘制:绘制的层级深、页面复杂、刷新不合理,绘制任务太重,绘制内容耗时太长(>16ms)。
- 数据处理:导致这种卡顿场景的原因是数据处理量太大,一般分为三种情况,一是数据在处理UI线程,二是数据处理占用CPU高,导致主线程拿不到时间片,三是内存增加导致GC频繁,从而引起卡顿。
布局优化
一个页面的显示测量和绘制过程都是通过递归来完成的,多叉树遍历的时间与树的高度h有关,其时间复杂度O(h),如果层级太深,每增加一层则会增加更多的页面显示时间。
主要通过减少层级、减少测量和绘制时间,保持布局层级的扁平化、提高复用性三个方面入手
- 减少层级。合理使用RelativeLayout和LinerLayout,constrainLayout,合理使用Merge。
- 在不影响层级深度的情况下,使用LinearLayout而不是RelativeLayout。因为RelativeLayout会让子View调用2次onMeasure,LinearLayout ,Measure的耗时越长那么绘制效率就低。
- 尽可能少用wrap_content。wrap_content 会增加布局measure时计算成本,在已知宽高为固定值时,不用wrap_content 。
- 删除控件中无用的属性。
- 减少使用weight属性,会导致measure两次,Measure的耗时越长那么绘制效率就低。
- 使用merge减少层级
- 用TextView、EditText的灵活运用,减少控件的数量。
避免过度绘制
过度绘制是指在屏幕上的某个像素在同一帧的时间内被绘制了多次。某些像素区域被绘制了多次,从而浪费了多余的CPU以及GPU源。
避免过度绘制:
-
移除XML中非必须的背景,移除Window默认的背景、按需显示占位背景图片
-
自定义View优化。使用 canvas.clipRect()来帮助系统识别那些可见的区域,只有在这个区域内才会被绘制。
如何检测?
- 使用HierarchyViewer来查找Activity中的布局是否过于复杂
- 在开发者选项中打开Show GPU Overdraw选项进行观察是否存在过度绘制
- 使用TraceView来观察CPU执行情况
App启动优化
从心理学角度而言,越快的启动速度往往给用户以性能好,高效可靠的心理暗示,这就很容易让用户对其产生好感。通过ADB命令统计应用的启动时间:adb shell am start -W 首屏Activity。
安卓应用的启动方式分为三种:冷启动、暖启动、热启动,应用发生冷启动时,系统一定会执行:
- 开始加载并启动应用
- 应用启动后,显示一个空白的启动窗口(启动闪屏页)
- application的初始化
- 启动UI线程
- 创建Activity
- 导入视图(inflate view)
- 计算视图大小(onmesure view)
- 得到视图排版(onlayout view)
- 绘制视图(ondraw view)
暖启动
当应用中的 Activities 被销毁,但在内存中常驻时,应用的启动方式就会变为暖启动。相比冷启动,暖启动过程减少了对象初始化、布局加载等工作,启动时间更短。但启动时,系统依然会展示闪屏页,直到第一个 Activity 的内容呈现为止。
热启动
相比暖启动,热启动时应用做的工作更少,启动时间更短。热启动产生的场景很多,常见如:用户使用返回键退出应用,然后马上又重新启动应用。
优化方法:
- 使用Activity的windowBackground主题属性来为启动的Activity提供一个简单的drawable。等Activity加载完毕后,再去加载Activity的界面,而在Activity的界面中,我们将主题重新设置为正常的主题
@Override
protected void onCreate(Bundle savedInstanceState) {
setTheme(R.style.ThemeApp);
super.onCreate(savedInstanceState);
}
- 优化闪屏页的UI布局
- 启动加载逻辑优化。可以采用分布加载、异步加载、延期加载策略来提高应用启动速度。
- 数据初始化分析,加载数据可以考虑用线程初始化等策略。
- Application的onCreate(),attachBaseContext()中同样减少复杂和耗时的操作
- 将一张图片通过设置主题的方式显示为启动窗口.
- Application中主要做了各种三方组件的初始化,考虑异步初始化三方组件,不阻塞主线程。有时可能会出现WorkThread中尚未初始化完毕但MainThread中已经使用的错误,因此这种情况建议延迟到使用前再去初始化;比如说根据情况放到SplashActivity,WorkThread,Application中初始化
- 卡顿不能都靠异步来解决,错误的使用工程线程不仅不能改善卡顿,反而可能加剧卡顿。Thread、ThreadPoolExecutor、AsyncTask、HandlerThread、IntentService等都各有利弊;例如通常情况下ThreadPoolExecutor比Thread更加高效、优势明显,但是特定场景下单个时间点的表现Thread会比ThreadPoolExecutor好:同样的创建对象,ThreadPoolExecutor的开销明显比Thread大;
- 正确的开启线程也不能包治百病,例如执行网络请求会创建线程池,而在Application中正确的创建线程池势必也会降低启动速度;因此延迟操作也必不可少。
通过对traceview的详细跟踪以及代码的详细比对,我发现Phihome卡顿发生在:
部分数据库及IO的操作发生在首屏Activity主线程;
Application中创建了线程池;
首屏Activity网络请求密集;
工作线程使用未设置优先级;
信息未缓存,重复获取同样信息;
流程问题:例如闪屏图每次下载,当次使用;
以及其它细节问题:
执行无用老代码;
执行开发阶段使用的代码;
执行重复逻辑;
调用三方SDK里或者Demo里的多余代码;
启动总结
利用主题快速显示界面
** 异步初始化组件;**
** 梳理业务逻辑,延迟初始化组件、操作;**
** 正确使用线程;**
** 去掉无用代码、重复逻辑等。**
合理的刷新机制
有时数据的变化会促使页面刷新,但频繁的刷新会导致资源开销增加。如ListView、RecycleView。
- 减少刷新的次数和刷新的区域(只刷新需要更新的部分数据)
- 尽量避免后台有高的CPU线程运行
- ondraw方法不需要创建新的局部对象,这是因为ondraw方法是实时执行的,这样会产品大量的临时对象,导致占用了更多内存,并且使系统不断的GC。降低了执行效率。
- Ondraw方法不需要执行耗时操作,在ondraw方法里少使用循环,因为循环会占用CPU的时间。导致绘制不流畅,卡顿等等。
内存优化
Android系统会限制每个App可分配的最大内存。当内存不足时,会导致内存溢出,爆出OutOfMemoryError。当内存紧张时,会触发GC,占用cpu的时间片,因此频繁的GC会导致会导致系统卡顿。
优化内存空间
在移动设备上,由于物理设备的存储空间有限,因此使用最小内存对象或者资源可以减小内存开销,同时让GC 能更高效地回收不再需要使用的对象,让应用堆内存保持充足的可用内存,使应用更稳定高效地运行。常见做法如下:
-
对象引用。强引用、软引用、弱引用、虚引用四种引用类型,根据业务需求合理使用不同,选择不同的引用类型。
减少不必要的内存开销。注意自动装箱,增加内存复用,比如有效利用系统自带的资源、视图复用、对象池、Bitmap对象的复用(修改Bitmap的颜色格式)。还比如字符串资源,如果需要拼接,不要使用string,而是使用stringbuilder。 -
使用最优的数据类型。比如针对数据类容器结构,可以使用ArrayMap/SparseArray数据结构(Hashmap自动装箱意味着需要产生额外的对象,这对于内存的使用和垃圾回收产生影响.相对于HashMap来说每一次put会少创建一个对象(HashMapEntry)。),避免使用枚举类型,使用缓存Lrucache等等。
-
图片内存优化。可以设置位图规格,根据采样因子做压缩,用一些图片缓存方式对图片进行管理等
-
减少帧动画的使用,如果需要,通过SurfaceView实现
-
使用更轻量级的数据结构,比如ArrayMap/SparseArray
-
初始化时,尽可能指定HashMap的大小
-
合理的使用多进程
-
及时关闭service
-
在该onStop()里做释放资源(例如网络连接、注销广播等)的工作
-
相比于静态常量,枚举会有超过其两倍以上的内存开销,在android中需严格避免使用枚举
常见内存泄漏场景
-
资源性对象未关闭。比如Cursor、File文件等,往往都用了一些缓冲,在不使用时,应该及时关闭它们。
-
注册对象未注销。比如事件注册后未注销,会导致观察者列表中维持着对象的引用。
-
类的静态变量持有大数据对象。
-
非静态内部类的静态实例。
-
Handler临时性内存泄漏。如果Handler是非静态的,容易导致Activity或Service不会被回收。
-
容器中的对象没清理造成的内存泄漏。
-
错误的上下文引用,尽量使用Application。
-
单例造成内存泄露
-
Animation导致内存泄露,在Activity的ondestory()方法中调用Animation.cancle()进行停止,一些简单的动画我们可以通过自定义view来解决。
内存分析工具
- Memory Monitor
- LeakCanary
- Heap Viewer
- Android Device Monitor中的Application Tracker追踪内存分配信息
- 使用MAT分析Java堆
稳定
Android应用的稳定性定义很宽泛,影响稳定性的原因很多,比如内存使用不合理、代码异常场景考虑不周全、代码逻辑不合理等,都会对应用的稳定性造成影响。其中最常见的两个场景是:Crash和ANR,这两个错误将会使得程序无法使用,比较常用的解决方式如下:
-
提高代码质量。比如开发期间的代码审核,看些代码设计逻辑,业务合理性等。
-
代码静态扫描工具。常见工具有Android Lint、Findbugs、Checkstyle、PMD等等。
-
Crash监控。把一些崩溃的信息,异常信息及时地记录下来,以便后续分析解决。
-
Crash上传机制。在Crash后,尽量先保存日志到本地,然后等下一次网络正常时再上传日志信息
ANR问题
Android官方规定:activity如果5s内无响应事件(屏幕触摸事件或者键盘输入事件)。BroadcastReceiver如果在10s内无法处理完成。Service如果20s内无法处理完成。绝大多数就是因为线程阻塞导致的。
- Asynctask:为UI线程与工作线程之间进行快速处理的切换提供一种简单便捷的机制。适用于当下立即需要启动,但是异步执行的生命周期短暂的场景。
- HandlerThread:为某些回调方法或者等待某些执行任务的执行设置一个专属的线程,并提供线程任务的调度机制。
- ThreadPool:把任务分解成不同的单元,分发到各个不同的线程上,进行同时并发处理。
- IntentService:适合执行由Ui触发的后台任务。并可以把这些任务执行的情况通过一定的机制反馈给UI。
省电
节省流量和好点,减少cpu和Gpu的计算,优化网络访问。
计算优化,避开浮点运算等。
-
避免WaleLock使用不当。
-
需要进行网络请求时,我们需先判断网络当前的状态。如果非必要,可将网络请求放在wifi的环境下,因为wifi请求的耗电量远比移动数据的耗电量低的低。
-
使用Job Scheduler。(延迟非必须的操作到充电状态时,比如日志上报完全可以在夜间充电时完成,这点可以结合JobScheduler使用)
-
使用传感器采集数据时,一旦不需要记得取消注册.
如何检测:
- 手机选项中通过查看APP的电量消耗的统计数据
- 使用Battery Historian Tool来查看详细的电量消耗
网络优化
做好网络优化一方面可以提高体验,另一方面可以减少流量和电量的损耗.
如何优化:
- 根据实际场景设计缓存策略。比如说根据数据是否经常变化设置缓存时间。
- 减少数据传输量,对传输的数据做压缩.如果传输的是图片,需要选择合适的图片格式以及根据显示大小请求合适规格的图片.
- 某些情况下可以采用IP直连,一方面可以减少DNS解析时间,另一方面可以防止域名劫持
- 根据实际场景确定请求策略(比如说连接wifi和充电的情况下,修改请求频率)
- 根据情况选择图片格式,和图片色彩位数。
- 刷新数据时,尽可能使用局部刷新,而不是全局刷新
安装包小
应用的安装包越大,用户下载的门槛越高,特别是在移动网络情况下,用户在下载应用时,对安装包大小的要求更高,因此,减小安装包大小可以让更多用户愿意下载和体验产品。
减少安装包大小的常用方案:
- 代码混淆。使用ProGuard代码混淆器工具,它包括压缩、优化、混淆等功能。
- 资源优化。比如使用Android Lint删除冗余资源,资源文件最少化等。
- 图片优化。比如利用AAPT工具对PNG格式的图片做压缩处理,降低图片色彩位数等。使用Vertor Drawable替代png/jpeg,有选择的提供对应分辨率的图片资源
- 复用已经存在的图片,多用通过代码对已有图片进行变换的方式实现复用
- 避免重复功能的库,使用WebP图片格式等。
- 插件化。比如功能模块放在服务器上,按需下载,可以减少安装包大小。
- 减少so文件的数量,根据实际情况提供so文件
- 使用Gradle中的shrinkResource来将无用的代码和资源排除在APK安装包之外
- 减少不必要的依赖库/Jar,在满足需求的前提下优先选择体积小的.
使代码高效的建议
- 如果方法不需要访问某对像的字段,将该方法设置为静态,调用速度会提升15%~20%
- 对于常量使用 final static
- int的数组比Integer对象数组要好得多。两个平行的int数组要比一个(int,int)型的对象数组高效。这对于其他任何基本数据类型的组合都通用
- 循环数组结构的数据时,建议使用普通for循环,链表结构时使用增强for循环
- 避免使用浮点数,通常的经验是,在Android设备中,浮点数会比整型慢两倍
数据库操作方法的优化
- 尽量利用原生的SQL语句
- 当操作条数较多时,利用事务进行批处理
这样SQLite将把全部要执行的SQL语句先缓存在内存当中,然后等到COMMIT的时候一次性的写入数据库,这样数据库文件只被打开关闭了一次,效率自然大大的提高