Android 掉帧优化

对于传统的60刷新率手机来说,每16ms会发出一个VSync信号,复制CPU/GPU放在缓存中的图像,再通知CPU/GPU计算下一帧要显示的内容,再把刚复制的图像显示在屏幕上,这就是一个屏幕刷新周期。而如果在16ms内没有计算完毕的话,该帧就无法展示,屏幕进入下一个刷新周期,就产生了所谓的掉帧现象。

1. 掉帧监控 监控掉帧现象时,我们可以使用下方的adb命令,具体可见参考.

adb shell dumpsys gfxinfo <packageName>

复制

该命令展示的信息比较完整,如下所示。

Applications Graphics Acceleration Info:
Uptime: 275941522 Realtime: 391854346** Graphics info for pid 6887 [packageName] **Stats since: 275926453465347ns
Total frames rendered: 523   // 本次共收集了523帧的信息
Janky frames: 26 (4.97%)   // 有26帧的耗时超过16ms,掉帧率为4.97%
50th percentile: 5ms   // 50%的帧耗时在5ms以内
90th percentile: 8ms
95th percentile: 16ms
99th percentile: 20ms
Number Missed Vsync: 0   // 垂直同步失败的帧
Number High input latency: 259   // 处理input时间超时的帧数
Number Slow UI thread: 1   // 因UI线程上的工作导致超时的帧数
Number Slow bitmap uploads: 0   // 因bitmap的加载耗时的帧数
Number Slow issue draw commands: 0   // 因绘制导致耗时的帧数
Number Frame deadline missed: 1
HISTOGRAM: 5ms=346 6ms=72 7ms=31 .........   // 耗时0-5ms的帧有346......各种缓存......
Total GPU memory usage:40704248 bytes, 38.82 MB (36.77 MB is purgeable)
......

复制

如果只看掉帧率,可以用adb shell dumpsys gfxinfo| grep "Janky frames"命令。 如果想重新开始计算帧率信息,可以通过adb shell dumpsys gfxinforeset重置。

当然我们也可以通过可视化界面查看UI性能,打开"开发者选项"中的"GPU渲染模式分析",即可在屏幕上看到每一帧绘制时间的直方图,某个值越大,代表该帧绘制的时间越长。如下图所示,冷启动APP时有不少帧的绘制时间已经远远超过了16ms。

除了"GPU渲染模式分析",还有Android Studio中的CPU Profile用于查看APP运行时的方法调用栈,辅助开发人员定位热点方法并优化。我们来做个实验,在Demo中的onBindViewHolder()中添加Thread.sleep(5),使每次绑定ItemView都会多消耗5ms。

运行程序后打开Profile,可以看到CPU、MEMORY、NETWORK和ENERGY四个动态图表,点击CPU后,下方出现CPU Profile界面,如下所示,点击"record"即可开始记录,点击"stop"后得到这一段时间内的方法调用栈。

得到方法调用栈信息后,先从"Flame Chart"模式来看热点方法,很明显sleep函数耗时较多。

如果想要数字化的信息,可以通过"Top Down"模式查看每个方法及其子方法的耗时和百分比,分析时一般点击耗时占比高的方法查看它的子方法哪个耗时较多,再一步步追踪下去。 在我们的例子中,sleep()函数占总耗时的49.58%,是耗时最多的方法。

总结一下,CPU Profile为开发者提供了强大的分析工具,我们很容易定位APP运行时耗时多的方法,然后具体问题具体分析。当然CPU Profile不仅仅用于掉帧优化,有优化的地方就有它的身影,例如启动优化等。

2. 掉帧优化措施

① 正确使用缓存

关于mCachedViews: mCachedViews针对ItemView的position进行缓存。当一个Item滑出可视区域时,它会先被放入mCachedViews中;而当一个Item滑入可视区域时,Recycler也会优先去mCachedViews中查找。 根据这个特性,当用户频繁地上下滑动时,mCachedViews的利用率会较高。那么针对频繁上下滑动的场景,我们可以通过RecyclerView.setItemViewCacheSize(…)来增大mCachedViews的容量,这样Recycler更容易在mCachedViews中找到缓存,减少之后的onBindViewHolder()和onCreateViewHolder()调用。

关于RecyclerPool: RecyclerPool针对某个ViewType进行缓存,默认大小为5,但是对于某些场景这是远远不够的。试想一个能在可视区域展示n(n>>5)条数据的RecyclerView(如历史记录),当滑动的时候RecyclerPool的缓存明显不够,会不断地创建ViewHolder,很消耗性能。针对这种情况,可以通过RecyclerView.getRecycledViewPool().setMaxRecycledViews(int viewType, int max)增大特定ViewType的缓存容量。

如果多个RecyclerView的内容性质相同,例如在信息流中,多个Fragment中的Item类型相同。那么可以为它们设置同一个RecyclerPool(默认是1个RecyclerView创建一个RecyclerPool),通过RecyclerView.setRecycledViewPool(pool)设置即可。

② 优化onBindViewHolder()耗时

从RecyclerPool中取出的ViewHolder都会调用onBindViewHolder()加载数据,该方法是在主线程运行的,处理不当时很容易造成滑动卡顿。 当为ItemView设置点击监听时,不要在onBindViewHolder()中新建OnClickListener,这不仅会新建多余的对象消耗内存,也会增加onBindViewHolder()的耗时。可以让所有的Item共用一个监听器,然后根据具体的Item来处理事件。

平时重写的onBindViewHolder(ViewHolder holder, int pos)会更新ItemView的所有内容,如果想要局部更新,可以重写onBindViewHolder(ViewHolder holder, int pos, Listpayloads)。当ItemView更新时,调用Adapter.notifyItemChanged(position, payLoad)即可。具体可见参考5,通过这个方法解决了ItemView更新时图片闪烁的问题。

③ 布局优化 布局优化一个比较典型的优化项就是优化过度绘制,打开"开发者选项"中的"调试GPU过度绘制",就能看到屏幕上每个像素点在屏幕上绘制了多少次。

对过度绘制进行优化时,首先要考虑合适的控件容器,也就是Layout。虽然Google推出了约束布局ConstraintLayout,但是它性能上并不优秀,不建议使用。 其次要善用merge和ViewStub。merge用于减少布局层级,例如自定义ViewGroup时,可以用作为根布局。ViewStub是布局文件中的占位符,对于某些在特殊场景下才需要显示的控件,可以先用ViewStub代替,等到需要显示时再加载。

还有一个常见的优化项就是layout_weight,该属性可以很轻松地实现空间分配,但是也很容易成为性能瓶颈,能不用就不用。

④ measure()优化和减少requestLayout()调用

当RecyclerView宽高的测量模式都是EXACTLY时,onMeasure()方法不需要执行dispatchLayoutStep1()等方法来进行测量。而当RecyclerView的宽高不确定并且至少一个child的宽高不确定时,要measure两遍。 因此将RecyclerView的宽高模式都设置为EXACTLY有助于优化性能。

protected void onMeasure(int widthSpec, int heightSpec) {// ......if (mLayout.isAutoMeasureEnabled()) {final int widthMode = MeasureSpec.getMode(widthSpec);final int heightMode = MeasureSpec.getMode(heightSpec);mLayout.onMeasure(mRecycler, mState, widthSpec, heightSpec);final boolean measureSpecModeIsExactly =widthMode == MeasureSpec.EXACTLY && heightMode == MeasureSpec.EXACTLY;if (measureSpecModeIsExactly || mAdapter == null) {return;}// ......
}

复制

还有一个方法RecyclerView.setHasFixedSize(true)可以避免数据改变时重新计算RecyclerView的大小,来看一下方法注释。 注释上说,如果Adapter的变化不会影响RecyclerView的size,那么可以设置mHasFixedSize为true来避免Adapter改变时RecyclerView刷新整个Layout。也就是说,不管数据变成什么样,如果RecyclerView的宽高都不会变,那么设置这个属性为true。

/*** RecyclerView can perform several optimizations if it can know in advance that RecyclerView's* size is not affected by the adapter contents. RecyclerView can still change its size based* on other factors (e.g. its parent's size) but this size calculation cannot depend on the* size of its children or contents of its adapter (except the number of items in the adapter).* <p>* If your use of RecyclerView falls into this category, set this to {@code true}. It will allow* RecyclerView to avoid invalidating the whole layout when its adapter contents change.** @param hasFixedSize true if adapter changes cannot affect the size of the RecyclerView.*/
public void setHasFixedSize(boolean hasFixedSize) {mHasFixedSize = hasFixedSize;
}

复制

当Adapter调用onItemRangeChanged(), onItemRangeInserted(), onItemRangeRemoved(), onItemRangeMoved()这4个方法时会调用triggerUpdateProcessor(),当mHasFixedSize为true时,不会调用requestLayout()重新计算宽高。 注意:如果调用notifyDataSetChanged()还是会调用requestLayout()去计算宽高。

void triggerUpdateProcessor() {if (POST_UPDATES_ON_ANIMATION && mHasFixedSize && mIsAttached) {ViewCompat.postOnAnimation(RecyclerView.this, mUpdateChildViewsRunnable);} else {mAdapterUpdateDuringMeasure = true;requestLayout();}
}

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

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

相关文章

六、源NAT实验

学习防火墙之前&#xff0c;对路由交换应要有一定的认识 源NAT1.私网用户通过NAT No-PAT访问Internet2.私网用户通过NATP访问Internet3.私网用户通过Easy-IP访问Internet4.私网用户通过三元组NAT访问Internet5.双出口环境下私网用户通过NAPT访问Internet 源NAT ———————…

预览功能实现

需求&#xff1a;将后端返回来的文字或者图片和视频展示在页面上。 <!-- 预览 --><el-dialog title"预览" :visible.sync"dialogPreviewVisible" width"50%" append-to-body :close-on-click-modal"false" close"Previe…

微服务--03--OpenFeign 实现远程调用

提示&#xff1a;文章写完后&#xff0c;目录可以自动生成&#xff0c;如何生成可参考右边的帮助文档 文章目录 OpenFeign其作用就是基于SpringMVC的常见注解&#xff0c;帮我们优雅的实现http请求的发送。 RestTemplate实现了服务的远程调用 OpenFeign快速入门1.引入依赖2.启用…

js手写数组push(),unshift(),pop(),shift(),map()方法

目录 1、push() 2、unshift() 3、pop() 4、shift() 5、map() 1、push() Array.prototype.pushfunction(){for(let i0;i<arguments.length;i){this[this.length]arguments[i]}return this.length}const arr[1,2,3]console.log(arr.push(4,5,6)) 2、unshift() Array.prot…

Docke日常指令

本文针对ubuntu操作系统而言&#xff1a; 补充&#xff1a;1.XAhost命令是X服务器的访问控制工具&#xff0c;用来控制哪些X客户端能够在X服务器上显示。 2.容器与镜像之间的关系&#xff1a;镜像你可以把它看成Java中的类&#xff0c;而容器可以看做是类的实例化对象&#xf…

OpenCV快速入门【完结】:总目录——初窥计算机视觉

文章目录 前言目录1. OpenCV快速入门&#xff1a;初探2. OpenCV快速入门&#xff1a;像素操作和图像变换3. OpenCV快速入门&#xff1a;绘制图形、图像金字塔和感兴趣区域4. OpenCV快速入门&#xff1a;图像滤波与边缘检测5. OpenCV快速入门&#xff1a;图像形态学操作6. OpenC…

创建可以离线打包开发的uniapp H5项目

安装node环境 略 安装vue脚手架&#xff0c;在线 npm install -g vue/cli PS&#xff1a;vue-cli已进入维护模式&#xff0c;vue3最新脚手架使用npm init vuelatest安装&#xff0c;安装后使用create-vue替换vue指令&#xff0c;create-vue底层使用vite提升前端开发效率&…

Redis应用的16个场景

常见的16种应用场景: 缓存、数据共享分布式、分布式锁、全局 ID、计数器、限流、位统计、购物车、用户消息时间线 timeline、消息队列、抽奖、点赞、签到、打卡、商品标签、商品筛选、用户关注、推荐模型、排行榜. 1、缓存 String类型 例如&#xff1a;热点数据缓存&#x…

Redis 命令处理过程

我们知道 Redis 是一个基于内存的高性能键值数据库, 它支持多种数据结构, 提供了丰富的命令, 可以用来实现缓存、消息队列、分布式锁等功能。 而在享受 Redis 带来的种种好处时, 是否曾好奇过 Redis 是如何处理我们发往它的命令的呢&#xff1f; 本文将以伪代码的形式简单分析…

【git】工作中常用的命令

前言 一些工作学习中常用的git命令小合集 正文 git clone git clone 使用的账号密码 用https的链接&#xff0c;就是要用github/gitee对应的用户名和密码。 git clone 获取指定指定分支的指定commit版本 第一步&#xff1a; git clone [git-url] -b [branch-name] 第二步…

centos 显卡驱动安装(chatglm2大模型安装步骤一)

1.服务器配置 服务器系统:Centos7.9 x64 显卡:RTX3090 (24G) 2.安装环境 2.1 检查显卡驱动是否安装 输入命令:nvidia-smi(显示显卡信息) 如果有以下显示说明,已经有显卡驱动。否则需要重装。 2.2 下载显卡驱动 第一步:浏览器输入https://www.nvidia.cn/Downloa…

Python读取Ansible playbooks返回信息

一&#xff0e;背景及概要设计 当公司管理维护的服务器到达一定规模后&#xff0c;就必然借助远程自动化运维工具&#xff0c;而ansible是其中备选之一。Ansible基于Python开发&#xff0c;集合了众多运维工具&#xff08;puppet、chef、func、fabric&#xff09;的优点&#x…

浅谈C++中如何重载前置++/--与后置++/--

前置/–与后置/– C中的和–操作符存在前置式与后置式&#xff0c;最基本的&#xff0c;作为一名程序员&#xff0c;你应当了解它们实现的不同&#xff1a; i; i;以上两行代码如果单独使用&#xff0c;在功能上是一致的&#xff0c;它们都实现了让变量i加1的操作&#xff0c;…

nuxt、vue实现PDF和视频文件的上传、下载、预览

上传 上传页面 <el-form-item :label"(form.ququ3 1 ? 参培 : form.ququ3 2 ? 授课 : ) 证明材料" prop"ququ6"><PdfUpload v-model"form.ququ6" :fileType"[pdf, mp4, avi, ts]"></PdfUpload> </el-form-i…

python:使用for循环与while循环打印九九乘法表

python&#xff1a;使用for循环与while循环打印九九乘法表 在编程中&#xff0c;for循环和while循环是两种常用的循环结构&#xff0c;它们可以用来实现各种不同的功能和逻辑。其中&#xff0c;九九乘法表是一个经典的例子&#xff0c;可以用来展示for循环和while循环的使用方…

亚马逊,shein,temu如何避免爆品评分低被强制下架

近期&#xff0c;一些Temu卖家反映产品下架问题&#xff0c;无论是日出千单的爆品还是其他商品&#xff0c;都有可能面临下架的风险。这其中最主要的原因之一是产品质量问题&#xff0c;导致消费者差评较多&#xff0c;评分降至4.2分或4.0分以下时&#xff0c;平台可能会强制下…

EfficientViT:具有级联群体注意力的内存高效Transformer

EfficientViT: Memory Efficient Vision Transformer with Cascaded Group Attention 1、介绍2、使用 Vision Transformer 加快速度2.1 内存效率2.2 计算效率2.3 参数效率 3、Efficient Vision Transformer3.1 EfficientViT 构建模块3.3 EfficientViT 网络架构 4、实验5、结论 …

YOLOv8独家原创改进: AKConv(可改变核卷积),即插即用的卷积,效果秒杀DSConv | 2023年11月最新发表

💡💡💡本文全网首发独家改进:可改变核卷积(AKConv),赋予卷积核任意数量的参数和任意采样形状,为网络开销和性能之间的权衡提供更丰富的选择,解决具有固定样本形状和正方形的卷积核不能很好地适应不断变化的目标的问题点,效果秒殺DSConv 1)AKConv替代标准卷积进行…

如何在vs2019及以后版本(如vs2022)上添加 添加ActiveX控件中的MFC类

有时候我们在MFC项目开发过程中&#xff0c;需要用到一些微软已经提供的功能&#xff0c;如VC使用EXCEL功能&#xff0c;这时候我们就能直接通过VS2019到如EXCEL.EXE方式&#xff0c;生成对应的OLE头文件&#xff0c;然后直接使用功能&#xff0c;那么&#xff0c;我们上篇文章…

el-tabel实现拖拽排序

1、使用npm安装sortableJs插件 npm install sortablejs --save2、在需要使用的页面进行引入 import Sortable from sortablejs3、完整代码 <template><div class"home"><el-table :data"tableData" style"width: 100%"><…