草庐IT

史上最全Android性能优化方案解析

嘴巴吃糖了 2023-04-15 原文

Android中的性能优分为以下几个方面:

布局优化

网络优化

安装包优化

内存优化

卡顿优化

启动优化

……

一.布局优化

布局优化的本质就是减少View的层级。常见的布局优化方案如下:

  • 在LinearLayout和RelativeLayout都可以完成布局的情况下优先选择LinearLayout,可以减少View的层级,但是注意相同组件可能RelativeLayout绘制时间长

  • 使用 < include > 标签将常用的布局组件共同的部分抽取出来,以便复用。

  • 通过 < ViewStub > 标签来加载不常用的布局,延迟加载(需要的时候在activity中加载出来)

  • 使用 < Merge > 标签来减少布局的嵌套层次

二.绘制优化

绘制优化是指View的onDraw方法要避免执行大量的操作,这主要体现在两个方面:

1.onDraw中不要创建新的局部对象。

因为onDraw方法可能会被频繁调用,这样就会在一瞬间产生大量的临时对象,这不仅占用了过多的内存而且还会导致系统更加频繁gc,降低了程序的执行效率。

2.onDraw方法中不要做耗时的任务,

不能执行成千上万次的循环操作,尽管每次循环都很轻量级,但是大量的循环仍然十分抢占CPU的时间片,这会造成View的绘制过程不流畅。

按照Google官方给出的性能优化典范中的标准,View的绘制频率保证60fps是最佳的,这就要求每帧绘制时间不超过16ms(16ms = 1000/60),虽然程序很难保证16ms这个时间,但是尽量降低onDraw方法中的复杂度总是切实有效的。

三.网络优化

常见的网络优化方案如下:

  • 尽量减少网络请求,能够合并的就尽量合并

  • 避免DNS解析,根据域名查询可能会耗费上百毫秒的时间,也可能存在DNS劫持的风险。可以根据业务需求采用增加动态更新IP的方式,或者在IP方式访问失败时切换到域名访问方式。

  • 大量数据的加载采用分页的方式

  • 网络数据传输采用GZIP压缩

  • 加入网络数据的缓存,避免频繁请求网络

  • 上传图片时,在必要的时候压缩图片

四.安装包优化

  • 安装包优化的核心就是减少apk的体积,常见的方案如下:

  • 减少应用中不必要的资源文件,比如图片,在不影响APP效果的情况下尽量压缩图片,有一定的效果

  • 在使用了SO库的时候优先保留v7版本的SO库,删掉其他版本的SO库。原因是在2018年,v7版本的SO库可以满足市面上绝大多数的要求,可能八九年前的手机满足不了,但我们也没必要去适配老掉牙的手机。实际开发中减少apk体积的效果是十分显著的,如果你使用了很多SO库,比方说一个版本的SO库一共10M,那么只保留v7版本,删掉armeabi和v8版本的SO库,一共可以减少20M的体积。

  • res资源优化

    (1)只使用一套图片,使用高分辨率的图片。

    (2)UI设计在ps安装TinyPNG插件,对图片进行无损压缩。

    (3)svg图片:一些图片的描述,牺牲CPU的计算能力的,节省空间。使用的原则:简单的图标。

    (4)图片使用WebP(https://developers.google.com/speed/webp/)的格式(Facebook、腾讯、淘宝在用。)缺点:加载相比于PNG要慢很多。但是配置比较高。工具:http://isparta.github.io/

    (5)使用tintcolor(android - Change drawable color programmatically)实现按钮反选效果。

  • 代码优化

    (1)实现功能模块的逻辑简化

    (2)Lint工具检查无用文件将无用的资源列在“UnusedResources: Unused resources”,删除。

    (3)移除无用的依赖库。

  • lib资源优化

    (1)动态下载的资源。

    (2)一些模块的插件化动态添加。

    (3)so文件的剪裁和压缩。

  • assets资源优化

    (1)音频文件最好使用有损压缩的格式,比如采用opus、mp3等格式,但是最好不要使用无损压缩的音乐格式

    (2)对ttf字体文件压缩,可以采用FontCreator工具只提取出你需要的文字。比如在做日期显示时,其实只需要数字字体,但是使用原有的字体库可能需要10MB大小,如果只是把你需要的字体提取出来生成的字体文件只有10KB

  • 代码混淆。

  • 使用proGuard 代码混淆器工具,它包括压缩、优化、混淆等功能。

  • 插件化

  • 可将功能模块放服务器,需要用时再加载。

  • 7z极限压缩

五.Android内存优化

1.Android内存管理机制

Android应用都是在Android虚拟机上运行的,内存分配和垃圾回收都是由Android虚拟机来完成的。

2.常见的内存泄漏

其实内存泄漏的本质就是较长生命周期的对象引用了较短生命周期的对象。

2.1 内存泄露

内存泄漏原因:堆上分配的对象已经不会再使用,但是GC收集器无法对其进行回收,此对象被强应用所引用 。

  • 静态变量导致的内存泄漏

    解决办法:将内部类设为静态内部类或独立出来;使用context.getApplicationContext()。

  • 单例模式导致的内存泄漏

    解决办法:传参context.getApplicationContext()。

  • 属性动画导致的内存泄漏

    解决办法:在Activity.onDestroy()中调用Animator.cancel()停止动画。

  • Handler导致的内存泄漏

    解决办法:使用静态内部类+WeakReference弱引用;当外部类结束生命周期时清空消息队列。

  • 线程导致的内存泄漏

    解决办法:将AsyncTask和Runnable设为静态内部类或独立出来;在线程内部采用弱引用保存Context引用。

  • 资源未关闭导致的内存泄漏

    解决办法:在Activity销毁的时候要及时关闭或者注销。例如:

    ① BraodcastReceiver:调用unregisterReceiver()注销;

    ②Cursor,Stream、File:调用close()关闭;

    ③Bitmap:调用recycle()释放内存(2.3版本后无需手动)。

  • Adapter导致的内存泄漏

    详情:不使用缓存而只依靠getView() 每次重新实例化Item,会给gc制造压力。

    解决办法:在构造Adapter时使用缓存的convertView。

  • WebView导致的内存泄漏。

    详情:WebView比较特殊,即使是调用了它的destroy方法,依然会导致内存泄漏。

    解决办法:其实避免WebView导致内存泄漏的最好方法就是让WebView所在的Activity处于另一个进程中,当这个Activity结束时杀死当前WebView所处的进程即可,我记得阿里钉钉的WebView就是另外开启的一个进程,应该也是采用这种方法避免内存泄漏。

  • 集合类泄漏

    详情:比如全局map等有静态应用,最后没有做删除。

    解决办法:在onDestry时回收不需要的集合。

2.2 扩大内存

大厂的SDK可能内存泄漏会少一些,但一些小厂的SDK质量也就不太靠谱一些。那应对这种我们无法改变的情况,最好的办法就是扩大内存。

扩大内存通常有两种方法:

  • 一个是在清单文件中的Application下添加largeHeap="true"这个属性,另一个就是同一个应用开启多个进程来扩大一个应用的总内存空间。

  • 第二种方法其实就很常见了,比方说我使用过个推的SDK,个推的Service其实就是处在另外一个单独的进程中。

  • Android中的内存优化总的来说就是开源和节流,开源就是扩大内存,节流就是避免内存泄漏。

2.3 检测、分析内存泄漏的工具

  • MemoryMonitor:随时间变化,内存占用的变化情况

  • MAT:输入HRPOF文件,输出分析结果

  • a. Histogram:查看不同类型对象及其大小

  • b.DominateTree:对象占用内存及其引用关系

  • c.MAT使用教程

  • LeakCanary:实时监测内存泄漏的库(LeakCanary原理)

六.卡顿优化方案

  • 不要在主线程进行网络访问/大文件的IO操作

  • 绘制UI时,尽量减少绘制UI层次;减少不必要的view嵌套,可以用Hierarchy Viewer工具来检测,后面会详细讲;

  • 当我们的布局是用的FrameLayout的时候,我们可以把它改成merge,可以避免自己的帧布局和系统的ContentFrameLayout帧布局重叠造成重复计算(measure和layout)

  • 提高显示速度,使用ViewStub:当加载的时候才会占用。不加载的时候就是隐藏的,仅仅占用位置。

  • 在view层级相同的情况下,尽量使用 LinerLayout而不是RelativeLayout;因为RelativeLayout在测量的时候会测量二次,而LinerLayout测量一次,可以看下它们的源码;

  • 删除控件中无用的属性;

  • 布局复用.比如listView 布局复用

  • 尽量避免过度绘制(overdraw),比如:背景经常容易造成过度绘制。由于我们布局设置了背景,同时用到的MaterialDesign的主题会默认给一个背景。这时应该把主题添加的背景去掉;还有移除

  • XML 中非必须的背景

  • 自定义View优化。使用 canvas.clipRect()来帮助系统识别那些可见的区域,只有在这个区域内才会被绘制。也是避免过度绘制.

  • 启动优化,启动速度的监控,发现影响启动速度的问题所在,优化启动逻辑,提高应用的启动速度。比如闪屏页面,合理优化布局,加载逻辑优化,数据准备.

  • 合理的刷新机制,尽量减少刷新次数,尽量避免后台有高的 CPU 线程运行,缩小刷新区域。

七.耗电优化

耗电的原因其实很多,这里我就讲一下几种优化方案,优化方案的反面就是他的原因了,几种优化方案如下:

  • 合理的使用wake_lock锁,wake_lock锁主要是相对系统的休眠(这里就是为了省电,才做休)而言的,意思就是我的程序给CPU加了这个锁那系统就不会休眠了,这样做的目的是为了全力配合我们程序的运行。有的情况如果不这么做就会出现一些问题,比如微信等及时通讯的心跳包会在熄屏不久后停止网络访问等问题。所以微信里面是有大量使用到了wake_lock锁。

  • 使用jobScheduler2,集中处理一些网络请求,有些不用很及时的处理可以放在充电的时候处理,比如,图片的处理,APP下载更新等等;

  • 计算优化,避开浮点运算等。

  • 数据在网络上传输时,尽量压缩数据后再传输,建议用FlatBuffer序列化技术,这个比json效率高很多倍,不了解FlatBuffer,建议找资料学习一下。

针对“性能优化”这个要点,分享给大家一份《360°全方位Android性能优化解析》,这份学习手册将会带领大家一步一步深入探索Android的性能优化,让产品的性能从各个方面得到提升,希望大家喜欢。
这份资料一共有721页,4个大点,25个小章节,不仅仅有详细的底层原理的解析,还有专门的实践案例!

《360°全方位Android性能优化解析》

第一章 设计思想与代码质量优化

1.六大原则

  • 单一职责原则

  • 里氏替换原则

  • 依赖倒转原则

  • 接口隔离原则

  • ……

2.设计模式

  • 结构型模式:桥接模式、适配器模式、装饰器模式、代理模式、门面(外观)模式……

  • 创建型模式:建造者模式、单例模式、抽象工厂模式、工厂方法模式……

  • 数据结构:数组、栈、队列、链表、树……

  • 算法:排序算法、查找算法……

第二章 程序性能优化

1.启动速度与执行效率优化

  • 冷启动和热启动解析

  • APP 启动黑白屏解决办法

  • APP 卡顿问题分析及解决方案

  • 启动速度与执行效率优化之 StrictMode

  • ……

2.布局检测与优化

  • 布局层级优化

  • 过度渲染

  • ……

3.内存优化

  • 内存抖动和内存泄漏

  • 内存大户

  • Bitmap 内存优化

  • Profile 内存监测工具

  • Mat 大对象与泄漏检测

  • 耗电优化

  • 网络传输与数据存储优化网络传输与数据存储优化

  • APK 大小优化

  • 屏幕适配

  • ……

4.耗电优化

  • Doze&Standby

  • Battery Historian

  • JobScheduler

  • WorkManager

5.网络传输与数据存储优化

  • google 序列化工具 protobuf

  • 7z 极限压缩

  • ……

6.APK 大小优化

  • APK 瘦身

  • 微信资源混淆原理

  • ……

7.屏幕适配

  • 进行适配的原理

  • 屏幕分辨率限定符与 smallestWidth 限定符适配原理

  • 为什么选择 smallestWidth 限定符适配

  • 怎么适配其他 module

  • 常见问题处理

    ……

8.OOM 问题原理解析

  • adj 内存管理机制

  • JVM 内存回收机制与 GC 算法解析

  • 生命周期相关问题总结

  • Bitmap 压缩方案总结

  • ……

9.ANR 问题解析

  • AMS 系统时间调节原理

  • 程序等待原理分析

  • ANR 问题解决方案

  • ……

10.Crash 监控方案

  • Java 层监控方案

  • Nativie 层监控方案

  • ……

第三章 开发效率优化

  1. 分布式版本控制系统 Git

企业高效持续集成平台场景介绍

GIT 分布式版本控制系统

  • GIT 分支管理

  • ……

2.自动化构建系统 Gradle:

  • Gradle 与 Android 插件:gradle 与 android gradle 插件的关系、Gradle Transform API 的基本使用……

  • Gradle Transform API 的基本使用:什么是 Transform、Transform 的使用场景、Transform API 学习、输入的类型……

  • 自定义插件开发:Gradle 插件简介、开始准备、实践、自定义 Gradle 插件、buildSrc 模块方式……

  • 插件实战:多渠道打包、发版自动钉钉……

第四章 APP 性能优化实践

1.启动速度

  • 应用启动的一般流程

  • 冷启动和热启动

  • 启动速度的测量

  • 启动窗口优化

  • 线程优化

  • 系统调度优化

  • GC 优化

  • IO 优化

  • 资源重排

  • 主页布局优化

  • 类加载优化

  • 选择合适的启动框架

  • 减少 Activity 的跳转层次

  • 厂商优化

  • 后台保活

  • ……

  • 2.流畅度

  • 性能问题分析的一些工具和套路

  • 通过性能数据数据分析

  • Android 平台性能导致的性能案例

  • Android App 自身导致的性能问题

  • 低内存的数据特征和行为特征

  • 应用宝

  • 讯飞输入法无障碍服务导致的整机卡顿分析

  • 字节跳动:今日头条图文详情页秒开实践

  • ……

  • 3.抖音在 APK 包大小资源优化的实践

  • 图片压缩

  • webp 无侵入式兼容

  • 多 DPI 优化

  • 重复资源合并

  • shrinkResource 严格模式

  • 资源混淆(兼容 aab 模式)

  • ARSC 瘦身

  • ……

  • 4.优酷响应式布局技术全解析

  • 优酷APP响应式布局技术概述

  • 优酷APP响应式布局Android落地

  • 在分发场景的落地

  • 在消费场景的落地

  • 优酷APP响应式布局之测试方案

  • ……

  • 5.网络优化

  • 手机淘宝在网络的链路优化

  • 百度 APP 在网络深度优化的实践

  • ……

  • 6.手机淘宝双十一性能优化项目揭秘

  • 一秒法则的实现

  • 启动时间和页面帧率提升 20%

  • Android 手机内存节省50%

  • ……

  • 7.高德 APP 全链路源码依赖分析

  • 高德 APP 平台架构

  • 基础实现原理

  • 项目架构

  • 应用场景及实现原理

  • ……

  • 8.彻底干掉OOM的实战经验分享

  • 排查内存泄漏

  • 兜底策略

  • 内存峰值太高

  • 特大图排查优化

  • ……

  • 9.微信 Android终端内存优化实践

  • Activity 泄露检测

  • Bitmap 分配及回收追踪

  • Native 内存泄漏检测

  • 线程监控

  • 内存监控

  • ……

总结

性能优化不是更新一两个版本就可以解决的,是持续性的需求,持续集成迭代反馈。在实际的项目中,在项目刚开始的时候,由于人力和项目完成时间限制,性能优化的优先级比较低,等进入项目投入使用阶段,就需要把优先级提高,但在项目初期,在设计架构方案时,性能优化的点也需要提早考虑进去,这就体现出一个程序员的技术功底了。希望这份《360°全方位Android性能优化解析》可以给到你帮助。

文中资料可扫描下方CSDN官方认证卡片免费领取!

有关史上最全Android性能优化方案解析的更多相关文章

  1. Ruby 解析字符串 - 2

    我有一个字符串input="maybe(thisis|thatwas)some((nice|ugly)(day|night)|(strange(weather|time)))"Ruby中解析该字符串的最佳方法是什么?我的意思是脚本应该能够像这样构建句子:maybethisissomeuglynightmaybethatwassomenicenightmaybethiswassomestrangetime等等,你明白了......我应该一个字符一个字符地读取字符串并构建一个带有堆栈的状态机来存储括号值以供以后计算,还是有更好的方法?也许为此目的准备了一个开箱即用的库?

  2. ruby - 解析 RDFa、微数据等的最佳方式是什么,使用统一的模式/词汇(例如 schema.org)存储和显示信息 - 2

    我主要使用Ruby来执行此操作,但到目前为止我的攻击计划如下:使用gemsrdf、rdf-rdfa和rdf-microdata或mida来解析给定任何URI的数据。我认为最好映射到像schema.org这样的统一模式,例如使用这个yaml文件,它试图描述数据词汇表和opengraph到schema.org之间的转换:#SchemaXtoschema.orgconversion#data-vocabularyDV:name:namestreet-address:streetAddressregion:addressRegionlocality:addressLocalityphoto:i

  3. ruby - 用逗号、双引号和编码解析 csv - 2

    我正在使用ruby​​1.9解析以下带有MacRoman字符的csv文件#encoding:ISO-8859-1#csv_parse.csvName,main-dialogue"Marceu","Giveittohimóhe,hiswife."我做了以下解析。require'csv'input_string=File.read("../csv_parse.rb").force_encoding("ISO-8859-1").encode("UTF-8")#=>"Name,main-dialogue\r\n\"Marceu\",\"Giveittohim\x97he,hiswife.\"\

  4. ruby - 在 jRuby 中使用 'fork' 生成进程的替代方案? - 2

    在MRIRuby中我可以这样做:deftransferinternal_server=self.init_serverpid=forkdointernal_server.runend#Maketheserverprocessrunindependently.Process.detach(pid)internal_client=self.init_client#Dootherstuffwithconnectingtointernal_server...internal_client.post('somedata')ensure#KillserverProcess.kill('KILL',

  5. ruby-on-rails - 我更新了 ruby​​ gems,现在到处都收到解析树错误和弃用警告! - 2

    简而言之错误:NOTE:Gem::SourceIndex#add_specisdeprecated,useSpecification.add_spec.Itwillberemovedonorafter2011-11-01.Gem::SourceIndex#add_speccalledfrom/opt/local/lib/ruby/site_ruby/1.8/rubygems/source_index.rb:91./opt/local/lib/ruby/gems/1.8/gems/rails-2.3.8/lib/rails/gem_dependency.rb:275:in`==':und

  6. 安卓apk修改(Android反编译apk) - 2

    最近因为项目需要,需要将Android手机系统自带的某个系统软件反编译并更改里面某个资源,并重新打包,签名生成新的自定义的apk,下面我来介绍一下我的实现过程。APK修改,分为以下几步:反编译解包,修改,重打包,修改签名等步骤。安卓apk修改准备工作1.系统配置好JavaJDK环境变量2.需要root权限的手机(针对系统自带apk,其他软件免root)3.Auto-Sign签名工具4.apktool工具安卓apk修改开始反编译本文拿Android系统里面的Settings.apk做demo,具体如何将apk获取出来在此就不过多介绍了,直接进入主题:按键win+R输入cmd,打开命令窗口,并将路

  7. ruby - 用 YAML.load 解析 json 安全吗? - 2

    我正在使用ruby2.1.0我有一个json文件。例如:test.json{"item":[{"apple":1},{"banana":2}]}用YAML.load加载这个文件安全吗?YAML.load(File.read('test.json'))我正在尝试加载一个json或yaml格式的文件。 最佳答案 YAML可以加载JSONYAML.load('{"something":"test","other":4}')=>{"something"=>"test","other"=>4}JSON将无法加载YAML。JSON.load("

  8. ruby - 如何使用 Nokogiri 解析纯 HTML 表格? - 2

    我想用Nokogiri解析HTML页面。页面的一部分有一个表,它没有使用任何特定的ID。是否可以提取如下内容:Today,3,455,34Today,1,1300,3664Today,10,100000,3444,Yesterday,3454,5656,3Yesterday,3545,1000,10Yesterday,3411,36223,15来自这个HTML:TodayYesterdayQntySizeLengthLengthSizeQnty345534345456563113003664354510001010100000344434113622315

  9. Ruby 的数字方法性能 - 2

    我正在使用Ruby解决一些ProjectEuler问题,特别是这里我要讨论的问题25(Fibonacci数列中包含1000位数字的第一项的索引是多少?)。起初,我使用的是Ruby2.2.3,我将问题编码为:number=3a=1b=2whileb.to_s.length但后来我发现2.4.2版本有一个名为digits的方法,这正是我需要的。我转换为代码:whileb.digits.length当我比较这两种方法时,digits慢得多。时间./025/problem025.rb0.13s用户0.02s系统80%cpu0.190总计./025/problem025.rb2.19s用户0.0

  10. python - 帮我找到合适的 ruby​​/python 解析器生成器 - 2

    我使用的第一个解析器生成器是Parse::RecDescent,它的指南/教程很棒,但它最有用的功能是它的调试工具,特别是tracing功能(通过将$RD_TRACE设置为1来激活)。我正在寻找可以帮助您调试其规则的解析器生成器。问题是,它必须用python或ruby​​编写,并且具有详细模式/跟踪模式或非常有用的调试技术。有人知道这样的解析器生成器吗?编辑:当我说调试时,我并不是指调试python或ruby​​。我指的是调试解析器生成器,查看它在每一步都在做什么,查看它正在读取的每个字符,它试图匹配的规则。希望你明白这一点。赏金编辑:要赢得赏金,请展示一个解析器生成器框架,并说明它的

随机推荐