React-Native 动画优化
前言
动画对于客户端来说是非常重要的一部分,直接影响到应用的用户体验。前端对于动画优化通常使用CSS3样式来实现动画,以利用GPU加速特性。而React-Native由于渲染模式的不同,无法使用CSS样式的方式优化。
React-Native的实现
React-Native在动画方面有两个主要方式,一个是Animated,一个是LayoutAnimation。
Animated
Animated动画库的原理是由JavaScript来进行动画的计算,然后在每帧设置对应组件的style来实现动画过程(requestAnimationFrame)。
这个动画库的特点是非常灵活,因为所有的控制都是通过JavaScript实现的,动画补帧都是在JavaScript端完成,可以实现各种复杂动画效果,包括跟手动画等。
但是成也JavaScript败也JavaScript,JavaScript是单线程的,动画的特点是要保证流畅就要保证每秒60的帧率,也就是说如果在16ms内处理不完,势必要掉帧。而在实际业务中,播放动画的同时一般都会执行一些任务处理逻辑。比如数据拉取、数据计算,典型情况通常会伴有Loading动画,切换页面过场时伴随新页面数据加载等。在复杂场景下,由于业务逻辑的加入,动画处理的帧率就很难保证,用我们QA的话说就是卡的像个Web。。。
优点:
- 可实现复杂动画
- 非常灵活
- 可以兼容Web
缺点:
- 性能问题,掉帧、卡顿
LayoutAnimation
React-Native自带还有一种动画的实现方式,就是LayoutAnimation。这种方案是直接在Native实现一些动画效果,然后由JavaScript进行设置调用,由于整个动画过程交由Native处理,使得性能得以保证。不过LayoutAnimation的实现思路有点问题,他只是预置了几个动画效果,并且只能配置在create/update/delete的时候,需要自定义的动画、跟手动画等都没法实现。
优点:
- 性能良好
缺点:
- 动画效果简单,可适用范围较窄
尝试解决
给他特权
针对Animated的优化思路就是解决动画播放时JavaScript任务量的问题,保证动画循环达到60帧。一个典型的解决方案就是暂时停止所有同步任务的处理,等到动画执行完成之后再同步或异步的执行任务处理。比如在切页转场前暂停耗时操作(延迟Redux的dispatch过程等),转场动画结束后再进行同步或异步处理,把大的同步任务用setTimeout等方式拆成多个异步过程处理。
这种方案可以一定程度上解决动画卡顿的问题,但是有几个缺点:
- 任务总是要执行的,延迟处理会导致整个处理过程变长,响应用户操作速度变慢,影响用户体验。
- 治标不治本,复杂动画该卡还得卡。。。
优点:
- 改动较少,成本低,万一以后facebook改进了呢?比如在独立线程进行动画渲染计算?(需要解决数据同步问题,WebWorker也是个坑)
- 可以实现跟手动画,控制好JavaScript的任务量,效果还可以
所以此方案适用 动画相对简单、后续任务不复杂 的场景
原生优化
要从根本上解决问题,就需要让动画过程脱离JavaScript。所以就要开发原生组件来实现动画过程,这样可以充分利用机器性能,可以使用GPU进行动画渲染。还需要考虑灵活性,如果只能实现固定的效果那么使用场景就大打折扣。
然而,我们并不需要从头设计,OC、Android、WPF,都有优秀的动画API,我们还等什么?抄吧。
(So,欢迎使用 react-native-animation 支持iOS、Android,目前验证可行、能用、不完善、待重构,欢迎PR~)
举个小栗子:
<Animation.AnimationView
data={[{
type: 'Alpha',
from: 0,
to: 1,
duration: 500,
}]} autoplay={true}>
other views...
</Animation.AnimationView>
*Now support: 'Translate' | 'Rotate' | 'Scale' | 'Alpha'
Props:
export interface PropsDefine {
data: AnimationModel[]
style?: React.ViewStyle
autoplay?: boolean
autoclear?: boolean
onStart?: (view: AnimationView) => void
onEnd?: (view: AnimationView) => void
}
数据模型:
export interface AnimationModel {
name?: string
type: 'Translate' | 'Rotate' | 'Scale' | 'Alpha'
from?: number
to?: number
from2?: number
to2?: number
duration: number
startOffset?: number
interpolator?: 'Linear'
interpolatorData?: number
repeat?: number
}
效果演示:
iOS微粉App(android版兼容开发中),业务逻辑全React-Native实现。赞踩动画、弹幕动画、切页动画、回复/评论弹层动画已应用优化。
虽然这种方案可以解决可预测的动画得卡顿问题,缺点也很明显:
- 动画的过程是预设好的,所以无法实现跟手动画
优点:
- 动画过程交由Native处理,性能Max,JavaScript任务处理不影响动画效果,可并行
所以此方案适用 可预测的动画 的场景
结语
虽然React-Native标榜的就是性能更好,但也只是跟Web对比而言,跟Native比还是有不少差距的,跟Xamarin等跨平台方案对比性能也差距不小,所以在开发时还是要关注下性能问题,性能调优还是要花不少精力去解决的。
微粉iOS:

React-Native 动画优化的更多相关文章
- React Native 性能优化指南【全网最全,值得收藏】
2020 年谈 React Native,在日新月异的前端圈,可能算比较另类了.文章动笔之前我也犹豫过,但是想到写技术文章又不是赶时髦,啥新潮写啥,所以还是动笔写了这篇 React Native 性能 ...
- React Native动画总结
最近在使用react native进行App混合开发,相对于H5的开发,RN所提供的样式表较少,RN中不能使用类似于css3中的动画,因此,RN提供了Animated的API 1.写一个最简单的动画 ...
- React Native性能优化之可取消的异步操作
前沿 在前端的项目开发中,异步操作是一个不可获取的,从用户的角度来说,异步操作所带来的体验是美妙的,但有时候也会带来一些性能隐患.比如说:有一个异步请求还没有返回结果,但是页面却关闭了,这时由于异步操 ...
- 慢牛系列四:好玩的React Native
在上次随笔(系列三)中,我试着用RN实现了一个Demo,感觉很不错,当时遇到的问题这篇文章里基本都解决了,比如导航动画问题,这篇文章里主要介绍RN的动画,学会动画以后,各种小创意都可以实现了^^ 下面 ...
- React Native细节知识点总结<二>
1.关于React Native导出组件的export default和export的问题: 一个文件只能有一个export default,可以有多个export export class Temp ...
- 【独家】React Native 版本升级指南
前言 React Native 作为一款跨端框架,有一个最让人头疼的问题,那就是版本更新.尤其是遇到大版本更新,JavaScript.iOS 和 Android 三端的配置构建文件都有非常大的变动,有 ...
- 📝 没 2 年 React Native 开发经验,你都遇不到这些坑
如果你喜欢我的文章,希望点赞 收藏 评论 三连支持一下,谢谢你,这对我真的很重要! React Native 开发时,如果只是写些简单的页面,基本上按着官方文档 reactnative.dev就能写出 ...
- H5、React Native、Native应用对比分析
每日更新关注:http://weibo.com/hanjunqiang 新浪微博!iOS开发者交流QQ群: 446310206 "存在即合理".凡是存在的,都是合乎规律的.任何新 ...
- 《React Native 精解与实战》书籍连载「React 与 React Native 简介」
此文是我的出版书籍<React Native 精解与实战>连载分享,此书由机械工业出版社出版,书中详解了 React Native 框架底层原理.React Native 组件布局.组件与 ...
- 1000 千米高空俯瞰 React Native
一.历史:React Native 从开始到现在 React Native 的定位是通过 React 构建原生 App: A framework for building native apps wi ...
随机推荐
- 浅谈 Fragment 生命周期
版权声明:本文为博主原创文章,未经博主允许不得转载. 微博:厉圣杰 源码:AndroidDemo/Fragment 文中如有纰漏,欢迎大家留言指出. Fragment 是在 Android 3.0 中 ...
- ASP.NET MVC 请求路径相关参数的获取
Request.ApplicationPath / Request.CurrentExecutionFilePath /Home/Index Request.FilePath /Home/Index ...
- 谈一下关于CQRS架构如何实现高性能
CQRS架构简介 前不久,看到博客园一位园友写了一篇文章,其中的观点是,要想高性能,需要尽量:避开网络开销(IO),避开海量数据,避开资源争夺.对于这3点,我觉得很有道理.所以也想谈一下,CQRS架构 ...
- nodejs之get/post请求的几种方式
最近一段时间在学习前端向服务器发送数据和请求数据,下面总结了一下向服务器发送请求用get和post的几种不同请求方式: 1.用form表单的方法:(1)get方法 前端代码: <form act ...
- 游戏编程系列[1]--游戏编程中RPC协议的使用[3]--体验
运行环境,客户端一般编译为.Net 3.5 Unity兼容,服务端因为用了一些库,所以一般为4.0 或往上.同一份代码,建立拥有2个项目.客户端引用: WindNet.Client服务端引用: OpL ...
- 有朋友问了数据库ID不连续,怎么获取上一篇和下一篇的文章?(不是所有情况都适用)
呃 (⊙o⊙)…,逆天好久没写SQL了,EF用的时间长了,SQL都不怎么熟悉了......[SQL水平比较菜,大牛勿喷] 方法很多种,说个最常见的处理 因为id是自增长的,所以一般情况下下一篇文章的I ...
- 【翻译】MongoDB指南/CRUD操作(一)
[原文地址]https://docs.mongodb.com/manual/ MongoDB CRUD操作(一) 主要内容:CRUD操作简介,插入文档,查询文档. CRUD操作包括创建.读取.更新和删 ...
- 在 C# 里使用 F# 的 option 变量
在使用 C# 与 F# 混合编程的时候(通常是使用 C# 实现 GUI,F#负责数据处理),经常会遇到要判断一个 option 是 None 还是 Some.虽然 Option module 里有 i ...
- CocoaPods的安装、使用、以及遇到的问题
CocoaPods是什么? 当你开发iOS应用时,会经常使用到很多第三方开源类库,比如JSONKit,AFNetWorking等等.可能某个类库又用到其他类库,所以要使用它,必须得另外下载其他类库,而 ...
- Python处理Excel表格
同事小王今天说他有一个Excel表格,表格如下,一列是姓名,一列是电话号码,总共有大概2000行数据. 有的姓名占了一行,有的占了两行,还有一些占了三行的.如下图: 他问我可不可以全部统一成一行,而且 ...