android 系统源码挖掘之Animator性能优化

2023-11-04 01:20

本文主要是介绍android 系统源码挖掘之Animator性能优化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

作者:zjw-swun

博客:https://juejin.im/user/58994f1d128fe1006cbfe0da

声明:本文由作者 zjw-swun 授权发布,未经原作者允许请勿转载


今天在看FragmentManager源码的时候看见了一段关于优化动画性能的代码,经过真机实测,发现确实达到了不错的性能优化效果,分享给大家

1. 优化前后效果对比图

前提: 手机为真机, 魅族MX5E, 系统 5.0( api 26的模拟器下看GPU 条形图不知道为什么优化代码反而不如不优化的,可能是没有硬件加速导致的吧)


优化前效果如下

640?wx_fmt=jpeg

优化后效果如下

640?wx_fmt=jpeg

经过对比发现,确实性能优化不少

2. 从android源码扣出来的优化动画的关键代码以及测试代码

640?wx_fmt=png

我这边给出我扣出来的代码(kotlin版本)

 1class AnimateOnHWLayerIfNeededListener(private var mView: View?) : Animator.AnimatorListener {
2        private var mShouldRunOnHWLayer = false
3
4
5        override fun onAnimationStart(animation: Animator) {
6            mShouldRunOnHWLayer = shouldRunOnHWLayer(mView, animation)
7            if (mShouldRunOnHWLayer) {
8                mView!!.setLayerType(View.LAYER_TYPE_HARDWARE, null)
9            }
10        }
11
12        override fun onAnimationEnd(animation: Animator) {
13            if (mShouldRunOnHWLayer) {
14                mView!!.setLayerType(View.LAYER_TYPE_NONE, null)
15            }
16            mView = null
17            animation.removeListener(this)
18        }
19
20        override fun onAnimationCancel(animation: Animator) {
21
22        }
23
24        override fun onAnimationRepeat(animation: Animator) {
25
26        }
27
28        fun shouldRunOnHWLayer(v: View?, anim: Animator?)Boolean {
29            return if (v == null || anim == null) {
30                false
31            } else v.layerType == View.LAYER_TYPE_NONE
32                    && v.hasOverlappingRendering()
33                    && modifiesAlpha(anim)
34        }
35
36
37        private fun modifiesAlpha(anim: Animator?)Boolean {
38            if (anim == null) {
39                return false
40            }
41            if (anim is ValueAnimator) {
42                val valueAnim = anim as ValueAnimator?
43                val values = valueAnim!!.values
44                for (i in values.indices) {
45                    if ("alpha" == values[i].propertyName) {
46                        return true
47                    }
48                }
49            } else if (anim is AnimatorSet) {
50                val animList = anim.childAnimations
51                for (i in animList.indices) {
52                    if (modifiesAlpha(animList[i])) {
53                        return true
54                    }
55                }
56            }
57            return false
58        }
59    }

java 版本如下

 1static class AnimateOnHWLayerIfNeededListener implements Animator.AnimatorListener {
2        private boolean mShouldRunOnHWLayer = false;
3        private View mView;
4        public AnimateOnHWLayerIfNeededListener(final View v) {
5            if (v == null) {
6                return;
7            }
8            mView = v;
9        }
10
11        @Override
12        public void onAnimationStart(Animator animation) {
13            mShouldRunOnHWLayer = shouldRunOnHWLayer(mView, animation);
14            if (mShouldRunOnHWLayer) {
15                mView.setLayerType(View.LAYER_TYPE_HARDWARE, null);
16            }
17        }
18
19        @Override
20        public void onAnimationEnd(Animator animation) {
21            if (mShouldRunOnHWLayer) {
22                mView.setLayerType(View.LAYER_TYPE_NONE, null);
23            }
24            mView = null;
25            animation.removeListener(this);
26        }
27
28        @Override
29        public void onAnimationCancel(Animator animation) {
30
31        }
32
33        @Override
34        public void onAnimationRepeat(Animator animation) {
35
36        }
37
38        static boolean shouldRunOnHWLayer(View v, Animator anim) {
39            if (v == null || anim == null) {
40                return false;
41            }
42            return v.getLayerType() == View.LAYER_TYPE_NONE
43                    && v.hasOverlappingRendering()
44                    && modifiesAlpha(anim);
45        }
46
47        static boolean modifiesAlpha(Animator anim) {
48            if (anim == null) {
49                return false;
50            }
51            if (anim instanceof ValueAnimator) {
52                ValueAnimator valueAnim = (ValueAnimator) anim;
53                PropertyValuesHolder[] values = valueAnim.getValues();
54                for (int i = 0; i < values.length; i++) {
55                    if (("alpha").equals(values[i].getPropertyName())) {
56                        return true;
57                    }
58                }
59            } else if (anim instanceof AnimatorSet) {
60                List<Animator> animList = ((AnimatorSet) anim).getChildAnimations();
61                for (int i = 0; i < animList.size(); i++) {
62                    if (modifiesAlpha(animList.get(i))) {
63                        return true;
64                    }
65                }
66            }
67            return false;
68        }
69    }

测试代码 xml如下 ,布局很简单5个宽高全屏幕的View,然后做alpha动画

640?wx_fmt=png

640?wx_fmt=png

重点就是alphaAnimation.addListener(AnimateOnHWLayerIfNeededListener(v))这句代码,就是用了FragmentManager源码中扣出来的动画优化代码。

3. 扩展总结

上述代码解决了alpha动画的优化,那么如果是scale,x,y移动动画呢,同理啦,就是把判断alpha的代码去掉就行了,核心其实就是动画开始前启用离屏缓冲,也就是mView.setLayerType(View.LAYER_TYPE_HARDWARE, null);,然后动画结束的时候,关闭离屏缓冲,也就是mView.setLayerType(View.LAYER_TYPE_NONE, null);,很多朋友搞不清楚,硬件加速和View.LAYER_TYPE_HARDWARE的关系,我这里说一下,android 4.0以后所有页面默认全部开启硬件加速,View树无特殊情况,LayerType是View.LAYER_TYPE_NONE,LAYER_TYPE_HARDWARE这个叫做硬件层面的离屏缓冲(学过java swing的同学应该知道一个叫双缓冲的东西,LAYER_TYPE_HARDWARE就是使用了硬件做双缓冲),LAYER_TYPE_HARDWARE和硬件加速关系是当硬件加速开启的时候才能使用硬件离屏缓冲(硬件双缓冲)LAYER_TYPE_HARDWARE,而如果你指定View的LayerType为LAYER_TYPE_SOFTWARE 这个叫做软离屏缓冲(用内存做双缓冲),使用了LAYER_TYPE_SOFTWARE等于主动放弃了硬件加速,那为什么要主动放弃能够提高渲染性能的硬件加速呢?因为有一些canvas的操作不支持硬件加速,这些不支持的点你可以去android开发者文档官网找到。

在说一下,上述思路解决了属性(alpha,scale,x,y)动画优化,如果要使用动画改变控件的宽高的时候如何优化呢?很明显会卡死,原理就是会导致在极端的时间了全局的requestLayout 重新measure layout draw整个视图树,优化方法可以看到我之前的基于recyclerView add子view思想动态改变控件宽高 优化的文章,链接为:https://www.imooc.com/article/21993

好了,分享到此结束。喜欢就点个推荐吧。


征稿科技互联网,技术类文章

投稿方式:

发送文章链接地址到邮箱:748778890@qq.com

注明“投稿”,收到会回复哟


这篇关于android 系统源码挖掘之Animator性能优化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/342226

相关文章

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

Android里面的Service种类以及启动方式

《Android里面的Service种类以及启动方式》Android中的Service分为前台服务和后台服务,前台服务需要亮身份牌并显示通知,后台服务则有启动方式选择,包括startService和b... 目录一句话总结:一、Service 的两种类型:1. 前台服务(必须亮身份牌)2. 后台服务(偷偷干

在不同系统间迁移Python程序的方法与教程

《在不同系统间迁移Python程序的方法与教程》本文介绍了几种将Windows上编写的Python程序迁移到Linux服务器上的方法,包括使用虚拟环境和依赖冻结、容器化技术(如Docker)、使用An... 目录使用虚拟环境和依赖冻结1. 创建虚拟环境2. 冻结依赖使用容器化技术(如 docker)1. 创

Deepseek使用指南与提问优化策略方式

《Deepseek使用指南与提问优化策略方式》本文介绍了DeepSeek语义搜索引擎的核心功能、集成方法及优化提问策略,通过自然语言处理和机器学习提供精准搜索结果,适用于智能客服、知识库检索等领域... 目录序言1. DeepSeek 概述2. DeepSeek 的集成与使用2.1 DeepSeek API

CentOS系统Maven安装教程分享

《CentOS系统Maven安装教程分享》本文介绍了如何在CentOS系统中安装Maven,并提供了一个简单的实际应用案例,安装Maven需要先安装Java和设置环境变量,Maven可以自动管理项目的... 目录准备工作下载并安装Maven常见问题及解决方法实际应用案例总结Maven是一个流行的项目管理工具

Tomcat高效部署与性能优化方式

《Tomcat高效部署与性能优化方式》本文介绍了如何高效部署Tomcat并进行性能优化,以确保Web应用的稳定运行和高效响应,高效部署包括环境准备、安装Tomcat、配置Tomcat、部署应用和启动T... 目录Tomcat高效部署与性能优化一、引言二、Tomcat高效部署三、Tomcat性能优化总结Tom

解读Redis秒杀优化方案(阻塞队列+基于Stream流的消息队列)

《解读Redis秒杀优化方案(阻塞队列+基于Stream流的消息队列)》该文章介绍了使用Redis的阻塞队列和Stream流的消息队列来优化秒杀系统的方案,通过将秒杀流程拆分为两条流水线,使用Redi... 目录Redis秒杀优化方案(阻塞队列+Stream流的消息队列)什么是消息队列?消费者组的工作方式每

Android kotlin语言实现删除文件的解决方案

《Androidkotlin语言实现删除文件的解决方案》:本文主要介绍Androidkotlin语言实现删除文件的解决方案,在项目开发过程中,尤其是需要跨平台协作的项目,那么删除用户指定的文件的... 目录一、前言二、适用环境三、模板内容1.权限申请2.Activity中的模板一、前言在项目开发过程中,尤

Go中sync.Once源码的深度讲解

《Go中sync.Once源码的深度讲解》sync.Once是Go语言标准库中的一个同步原语,用于确保某个操作只执行一次,本文将从源码出发为大家详细介绍一下sync.Once的具体使用,x希望对大家有... 目录概念简单示例源码解读总结概念sync.Once是Go语言标准库中的一个同步原语,用于确保某个操