探析Android中的四类性能优化

news2024/11/26 12:42:51

作者:Yj家的孺子牛

流畅性优化

主线程模型

了解 Android 的流畅性优化之前,我们需要先了解Android的线程结构。在 Android 中,有一个主线程模型,其中所有的绘制以及交互都是在主线程中进行的,所以,当我们编写的某类代码是需要在主线程中运行的,那么这类代码必然会影响到我们的绘制等,继而影响到流畅性问题。

下图之中,展示了我们哪类代码操作代码是会影响到主线程。

  1. System Events(系统事件)

    例如需要关闭、开启某个页面,这些都是需要通知我们的系统的来完成的,这些就是系统事件。它会触发到主线程的耗时操作。

  2. Input Events(输入事件)

    输入事件并非指单纯的在输入框输入信息的事件,而是在手机上进行操作。因为手机本身可以被抽象的指代为一个输入输出的设备。在触摸屏上任意的点击都是一个输入事件,它都是在主线程中完成的

  3. Application(应用事件)

    我们APP的启动都是要经过应用事件,而各种生命周期的初始化相关的回调、初始化也是要经过它。例如各类启动的 SDK 都是放在 Application 类中初始化的,这都是要在主线程中进行的。

  4. Services(服务事件)

    这一点大家也许有点疑惑,后台服务为什么会是在主线程的呢?我们不是一般新开一个线程在后台执行耗时任务的么?上述的两点其实是不矛盾的,只是我们平时的理解也许有误,认为 Services 作为后台任务就是在子线程才对。但事实上, Services 是运行在主线程中的,即使它是在后台执行的,而我们需要执行耗时任务的时候,再在 Services 中创建子线程来执行这个耗时任务 。

  5. Alarm(定时事件)

    定时事件中,由于它默认子线程是不安全的,主线程才是安全的,所以是需要到主线程去执行的。

UI绘制 虽然是在主线程中的,但是我们的代码是无法去控制它的,这是系统进行控制的。我们只能输入需要绘制什么,但是不能影响其绘制的流程。

界面更新以及卡顿

界面更新的VSync

知晓主线程模型后,我们来看一下 界面是如何刷新

下面的一图是介绍 Android 中的 VSync 机制,VSync 我们可以简单的理解为是一种屏幕刷新的信号,它会每隔一段时间就会刷新一次屏幕。

**垂直同步(VSync):**当屏幕从缓冲区扫描完一帧到屏幕上之后,开始扫描下一帧之前,发出的一个同步信号,该信号用来切换前缓冲区和后缓冲区。

如下图,UI Draw 会在 16ms 就执行一次,所以我们的需要让我们的绘制流程在 16ms 内执行完才不会出现卡顿问题。

所以说,如果我们不影响界面的更新,让每次绘制流程都可以保持在16ms以下,那么就不会出现卡顿问题了。

三类主要的卡顿以及其原因

  • 输入事件无法即时响应

    这类卡顿主要表现为我们滑动屏幕的时候,屏幕根本无法响应。这是由于我们在主线程中执行了一段非常耗时的、与事件无关的代码,而由于这段代码还在执行,所以输入事件根本没法执行,也没法绘制出我们的滑动效果了。

  • 输入事件立刻响应,但其耗时较长

    这类卡顿中,输入事件是立即响应的了,但是由于我们又在输入事件的分发中做了很多计算逻辑等耗时的操作,所以还是会出现卡顿。其表现出来的效果是,滑动的时候一卡一卡的,这是由于输入事件的执行超过了16ms了,导致了丢帧的问题。

  • UI Draw 之外需要在主线程中执行的任务耗时长

    我们知道,主线程中不单只有 UI 的绘制任务,还有其他的例如 Application 任务也是需要放置在主线程中执行的。这些任务如果不做处理,也一股脑的放置在主线程中执行,那么也会导致其占用时间过长,使得 APP 出现丢帧的问题。

解决卡顿问题

我们现已经知道了三大类卡顿的原因,其归结来说的原因就是主线程中出现了不必要的耗时操作,导致最后主线程的UI绘制出现阻塞或者溢出。那么我们解决卡顿、做流畅性优化的方法也是很明确了,那就是让主线程尽量只做交互(Input Event)以及刷新(UI Draw)。当然很多需要在主线程中的代码是无法避免的,但是我们尽量使其缩小,让所有的耗时代码都在子线程中运行,这样子使得其减少丢帧,能够更加的连贯顺滑。

没有VSync会怎么样?

上述讲到了 VSync 的作用以及卡顿的原因和处理,我们知道 VSync 信号可以用于同步刷新页面。那若是没有 VSync 信号,我们的页面会出现什么样的问题?

其实会出现画面撕裂。画面撕裂是什么呢?如下图中红色框中的图面,与上边出现了撕裂,这种就是画面撕裂。其出现的原因是上一帧的页面还在绘制中,下一帧页面就继续占用资源绘制了,所以会出现几帧的页面同时绘制在一个页面中,出现了画面撕裂感。

那为何映入 VSync 信号之后就不会有画面撕裂问题了呢?

在显示一张图片的时候,其流程为:GPU进⾏渲染—>帧缓存区⾥ —>显示控制器—>读取帧缓存区信息(位图) —> 数模转化(数字信号处—>模 拟型号) —>(逐⾏扫描)显示。 正常的情况下,显示器完全显示完一帧后,帧缓存区更新一帧,这样便不会有撕裂问题,但事实并非如此。 当显卡输出帧的速度比显示器快,显示器的处理速度跟不上显卡,在显示器处理显卡丢过来的第1帧的时候,第2帧就又到了(帧缓存区已更新),导致同一个画面同时出现1、2两帧,撕裂就产生了

在没有 VSync 信号的时候,一旦GPU渲染完后就会交由屏幕去将其绘制出来,那么 CPU 和 GPU 处理的事件有长有短,一旦两帧出现绘制冲突,就会出现画面撕裂问题了。

所以说,解决画面撕裂的核心是决定好数据的交换时机(绘制时机)由谁来控制。在绘制中,不应该是 CPU 处理写入之后就立即绘制,而应该是由屏幕渲染完一帧之后才去绘制绘制下一帧。然而屏幕不是控制器,它无法控制什么时候进行绘制,但是它可以传递 VSync 信号给 Android 系统,借助 VSync 信号,Android 就可以让 CPU 在新的一帧开始的时候立即处理显示问题了。

VSYNC 信号是由屏幕(显示设备)产生的,并且以 60fps 的固定频率发送给 Android 系统,Android 系统中的 SurfaceFlinger 接收发送的 VSYNC 信号。VSYNC 信号表明可对屏幕进行刷新而不会产生撕裂。

图片中未对 VSync 信号进行处理,导致出现卡顿问题。

资源优化

资源是什么?

资源指的是:Android 手机的软件和硬件资源,通俗意义上应用依赖于移动端的有限资源和系统规定的数值。例如:功耗、存储、流量、系统参数、CPU、内存等。

我们对上述的资源进行优化的时候,其并非无关联的、相互隔离的,例如流量消耗大,那么功耗消耗也会大,内存和存储的消耗也会大。但是我们需要优化某一个点的时候,是需要聚焦于其中这一点进行优化的。

当然,我们使用的设备也是在不断地优化的,但是并非说等着硬件的优化,我们的软件能跑就行。对于资源优化,我们追求的是利用最小的资源达到最好的效果,这是很有挑战性,对自我提升也很高的事情。

Android 能做哪些资源优化

上图中展示了对于当下技术特征的满意度,我们会发现上述的点其实都不算高。无论是内存/存储、电量或者是流量等等方面都是需要我们继续优化的点。而右图指的是如果后台的进程如果很多的话,使得内存占用很多,导致前台的APP也会收到内存的限制变得卡顿。

上图中对音量的优化其实收益也是很大的。例如当下的自媒体平台抖音,它就是在端侧进行了音量优化,取得了很大的收益。因为每一个人拍视频的音量和背景嘈杂度都是不一样的,所以我们需要在上传视频到平台的时候进行音量优化,对各种音量进行优化、拟合到同一水平,这样子可以使得用户在打开不同视频的时候,视频的音量不会影响到用户的体验。

而对于亮度的优化,可以使得我们的功耗降低,节省手机有限的电量。上面左图可以看出,深色模式对电量的影响是很显著的。而同时在优化亮度的时候,我们也可以找出功耗与体验的平衡点,使得功耗低的时候,我们的体验感也是最好的。这一点可以在右图中看出。

稳定性优化

对于稳定性优化,我们主要讲的点是 ANR,而其他的崩溃等则是需要尽可能的抹去,这里暂不做分析。ANR 是应用长时间处于阻塞状态的时候,系统会触发 ANR ,然后系统会询问是否需要强制退出或者继续等待。

ANR 存在的的原因是我们不能因为一个程序出现问题就使得整个手机无法使用。所以需要 ANR 来将错误的程序退出。而我们对其优化则需要尽可能的减少这种情况的出现,使得用户被打断,这是一种很不好的体验。

诊断 ANR 时需要考虑以下几种常见模式:

  1. 应用在主线程上非常缓慢地执行涉及 I/O 的操作。
  2. 应用在主线程上进行长时间的计算。
  3. 主线程在对另一个进程进行同步 binder 调用,而后者需要很长时间才能返回。
  4. 主线程处于阻塞状态,为发生在另一个线程上的长操作等待同步的块。
  5. 主线程在进程中或通过 binder 调用与另一个线程之间发生死锁。主线程不只是在等待长操作执行完毕,而且处于死锁状态。

系统级优化

对于 Android 系统,Google 本身也对其系统做了很多重大的优化,这些优化使得 Android 系统很多的问题得以解决,很多使用体验变得更好。而这些优化的点,我们可以作为今后的思路,让我们对 APP 的优化有更广阔的思考,让我们能发掘出更多的优化点和对其有更加深刻的思考

对于业务开发中,我们很多时候都是使用到 Application 和 FrameWork 层,但是需要真正做好性能优化,我们需要了解到很多的 核心库 Kernel 层的原理,这对我们自身来说,也是正向的收益。


其实想要全面掌握好 Android 性能优化的话,这些知识点你必须要有所了解,如: 内存优化、网络优化、卡顿优化、存储优化……等,为了让大家一次都可以了解全,所以将其整合成名为《Android 性能优化核心知识点手册》,大家可以参考下:

《APP 性能调优进阶手册》:https://qr18.cn/FVlo89

启动优化

内存优化

UI优化

网络优化

Bitmap优化与图片压缩优化

多线程并发优化与数据传输效率优化

体积包优化

《Android 性能调优核心笔记汇总》:https://qr18.cn/FVlo89

《Android 性能监控框架》:https://qr18.cn/FVlo89

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/459617.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【LaTex】Elsevier投稿系统到底何时整顿?‘expl3.sty‘ aborted!

前言 两年前,我在投稿Elsevier旗下的Knoeldeg-based systems时就被这个投稿系统整得是头昏脑胀,直接肝爆。首先,第一次提交手稿时可以接受PDF,很方便。然而,后面大修时提交可编辑的源文件时给我狠狠的打脸了。记得当时…

快速入门量化交易

本文首发自「慕课网」,想了解更多IT干货内容,程序员圈内热闻,欢迎关注"慕课网"! 原作者:袁霄|慕课网讲师 近来“量化交易”这个词听得越来越频繁,多数人对量化交易的第一印象是“高大上的技术”…

堆的原理解析

看这篇文章需要对比较器有一定的了解,可以看我的这篇文章: 认识比较器_鱼跃鹰飞的博客-CSDN博客 堆的实际存储方式是数组,但是脑海中应该把他想象成一种树的结构 依次加入下标0-8的9个数(添加过程中会不断的和父节点大小进行比…

舰船交流电网绝缘监测及故障定位的研究及产品选型

摘要:交流电网和电气设备的绝缘状况直接影响舰船电力系统安全,其绝缘电阻的下降是一个不可避免的过程,成为了电网安全的严重隐患。电气设备绝缘材料的劣化过程是不可逆的,对舰船交流电网进行绝缘在线监测及快速定位绝缘故障支路&a…

浅谈:JVM垃圾回收

一、四种类加载器(双亲委托/全盘委托机制) 1.启动类加载器: 加载 Java 核心类库,无法被 Java 程序直接引用。 2.扩展类加载器: 加载 Java 的扩展库。Java 虚拟机的实现会提供一个扩展库目录。该类加载器在此目录里面查找并加载 Java 类。 3.系统类加载器: 它根据 Java 应用的类…

seleniumUI自动化登录失败案例重新尝试WhileTrue

一个用户每次登录失败,失败N次,无法进入下一url时,怎样会重新尝试N次重新登录呢 ? 我们可以使用wihile true判断,并使用currenturl判断,下面就介绍以下个人的方法 currenturlEGTconfigFile.driver.curren…

Opencv识别车牌

Opencv识别车牌 #encoding:utf8 import cv2 import numpy as np Min_Area 50 #定位车牌 def color_position(img,output_path): colors [#([26,43,46], [34,255,255]), # 黄色 ([100,43,46], [124,255,255]), # 蓝色 ([35, 43, 46], [77, 255, 255]) # 绿色 ] hsv cv2.cvtCo…

推荐 7 个超牛的 Spring Cloud 实战项目

个 把一个大型的单个应用程序和服务拆分为数个甚至数十个的支持微服务,这就是微服务架构的架构概念,通过将功能分解到各个离散的服务中以实现对解决方案的解耦。 关于微服务相关的学习资料不多,而 GitHub 上的开源项目可以作为你微服务之旅…

STM32平衡小车 mpu6050学习

MPU6050简介 MPU6050是一款性价比很高的陀螺仪,可以读取X Y Z 三轴角度,X Y Z 三轴加速度,还有内置的温度传感器,在姿态解析方面应用非常广泛。 二、硬件连接 由于采用IIC通信,最基本的只需要采用四根线就可以了。分别VCC,GND,SCL,SDA连接到单片机 SCL-----PB6 SDA---…

23种设计模式之观察者模式(黑马程序员)

观察者模式 一、概述二、结构三、实现四、总结在最后 一、概述 观察者模式又被称为发布-订阅模式(Publish/Subscribe)模式,它定义了一种一对多的依赖关系,让多个观察者对象同时监听某一个主题对象。这个主题对象在状态发生变化时,会通知所有…

java 键值对详解及java键值对代码

在 Java中,对象可以理解为一个列表。这个列表里面的每个元素都是一个“键”,而每个“键”都是一个值。 键值对的概念,并不是在 Java中第一次出现,在 java 1.x中就已经有了。那时候它的意思是在一个命名空间中建立两个名字相同的对…

利用三维CNN对阿尔茨海默病进行多模态研究

文章目录 Is a PET All You Need? A Multi-modal Study for Alzheimer’s Disease Using 3D CNNs摘要方法实验结果讨论结论 Is a PET All You Need? A Multi-modal Study for Alzheimer’s Disease Using 3D CNNs 摘要 提出了一个系统评估多模态dnn的框架重新评估基于FDG-P…

Android-实现一个登录页面(kotlin)

准备工作 首先,确保你已经安装了 Android Studio。如果还没有安装,请访问 Android Studio 官网 下载并安装。 前提条件 - 安装并配置好 Android Studio Android Studio Electric Eel | 2022.1.1 Patch 2 Build #AI-221.6008.13.2211.9619390, built …

如何使用命令行添加配置码云仓库SSH秘钥-git仓库也一样

使用命令行添加配置码云仓库SSH秘钥 为什么要如何使用命令行添加配置码云仓库SSH秘钥?生成密钥你可以按如下命令来生成 sshkey:可以参考下图执行指令 添加密钥登录你的码云,鼠标移入头像,设置。点击 SSH公钥,打开配置页面&#x…

新型数字智慧城市综合趋势解决方案(ppt可编辑)

本资料来源公开网络,仅供个人学习,请勿商用,如有侵权请联系删除 新型智慧城市解决方案总体架构 新型智慧城市顶层规划(咨询)服务概述 服务定义:提供面向城市及其产业的智慧化咨询服务,涵盖需求…

linux——进程的概念与状态

大家好,我是旗帜僵尸。今天我将带领大家学习进程的概念。 本篇文章将继续收录于我的linux专栏中,若想查看关于linux其它知识的文章也可以点击右方链接。旗帜僵尸——linux 文章目录 一、进程概念冯诺依曼体系结构OS(操作系统Operator System&…

突破传统监测模式:业务状态监控HM的新思路

作者:京东保险 管顺利 一、传统监控系统的盲区,如何打造业务状态监控。 在系统架构设计中非常重要的一环是要做数据监控和数据最终一致性,关于一致性的补偿,已经由算法部的大佬总结过就不在赘述。这里主要讲如何去补偿&#xff…

电子阅读器市场角力,AI成为关键变量

配图来自Canva可画 近年来,随着国家“书香型社会”建设政策的出台,公众的阅读需求正在逐年增加,各类读书产品和读书活动,也如同雨后春笋般涌现,人们的阅读体验日益得到丰富。比如,昨天世界读书日举行的“不…

Photoshop在启动时出现读取计算机特定首选项时出错,或者提示暂存盘已满导致打不开该如何处理

上午还能用,下午打开Photoshop时就报此错误 点击确定后,出现下图错误 首先,先试试删除设置文件。在长按shiftctrlalt的情况下用鼠标右键点击Photoshop图标,点击打开(此间别松手),就会出现下图&a…

“智慧赋能 强链塑链”|工程物资供应链管理中的数字化应用

工程项目中的供应链管理至关重要 工程建设行业是国民经济的重要支柱之一,虽然在总产值上持续保持增长态势,但近年来行业的利润总额增速已连续多年呈现下降趋势。究其原因,可以大体从两个方面来看:一是行业盈利能力出现下降&#x…