使用React Native的一个重要原因就是达到60FPS的刷新,这看起来跟本地APP是一样的。在可能的情况下,我们尽量完善ReactNative的性能,使你只关注APP的逻辑,而可以不用管性能的优化。但是有的地方,我们还没有关注到。同样,跟本地代码(Object c)一样,我们不能确定哪种方式是最好的,所以还需要你手动干预。
这个指南的目的是教会你一些基础知识,以帮助您解决性能的问题,并且,讨论常见的一些性能问题以及解决的方法。
在你的祖父辈,它们一般把视频称为”移动的图片”的一个原因是:视频中逼真的动作是以固定的速度,快速改变静态的图片创造出来一个错觉。这里的每一个图片就一个是帧。每一秒中图片的数量(帧),它直接影响了视频的真时性(或者app中的用户界面). iOS的设备是每秒60帧。它也就给了你以及UI系统大约16.67毫秒的时间,来完成生成一张静态图片(帧)的所有工作。如果你不能在16.67ms内完成,则你将会丢失一个帧,UI会出现无响应状态。
在你的应用程序中,打开Deveroper菜单,并切换到fps监视器,你会发现有两种不同的帧速率。
许多的React Native应用,你的业务逻辑都是运行在Javascript线程上的。比如React 应用的生命周期(mount, update), API调用,触摸事件的处理等。并且在每一帧失效前, 更新原生支持的视图(Views)是被成批处理的,并且在每一次事件循环遍历结束后,发送给本地端。如果Javascript线程对于一个帧没有响应,则被认为删除这个帧。举例来说,在一个复杂的应用的root 组件中,你调用了this.setState
. 它会导致重新宣染子组件,这个过程很消耗资源。可以假设为它将花费200ms, 那么导致12帧丢弃(200ms / 16.67ms)。通过Javascript控制的作何动画,在这个过程会被冻结。如果作何计算超过100ms, 用户就可以感受到。
这通常发生在导航的切换中: 当你添加一个新的路由,Javascript线程需要读取这个场景所需要的所有组件,然后通过适当的命令发送给本地端,创建视图。这个过程会花费多个帧,引起卡顿,这是因为transition是由Javascript控制的。有此组件会在componentDidMount中做额外的计算,这可能会导航在transition卡顿的第二个原因.
另一个例子是响应触摸:如果你要做的任务在Javascript线程上跨越多个帧,你可能会注意到TouchableOpacity的延迟。这就是在Javascript 线程在忙的时候,不能处理从主线程发送过来的触摸事件的原因,所以会出现,native view调整了透明度,而又不对触摸事件做出响应。
许多人可能注意到,使用NavigationIOS的性能要比Navigator要好。这个原因是,transition(转场动画)的完成是在main thread中完成的。
同样,当Javascript被锁上时,你依然可以通过ScrollView向上和向下滚动,这是因为ScrollView存在于主线程中(虽然scroll event会向JS触发,但是对于滚动的发生是没有必要的).
当运行一个打包好的app, Console.log语句会引起很大的瓶颈。它会包含调试库redux-logger, 所以在打包前,确保删除了Console.log语句.
在dev模式中,会影响到Javascript线程的性能. 比如在运行时,向你提供警告和错误信息,检验propTypes。
上面提供过,Navigator动画是由Javascript线程控制。假设一个从右到左的场景转换, 既添加一个新页面:新的场景scene,是从右到左移动(-320 到 0),在这个转换过程的每一帧,Javascript thread需要将新的x位置发送给主线程。如果javascript 线程被冻结。它就不能做这些,那么这些帧就不会被更新,动画就变得断断续续。
一劳永逸的解决方案是将基于 Javascript的动画转变为基于main thread的动画。在上面的例子中,我们可能需要计算transition的每一个x偏移位置,然后发送给主线程,以一个优化的方式来执行。现在Javascript不需要在负责这个。
可是这中方案还没有实现,所以在这段期间,我们应该使用InteractionManager,为新的scene选择最少的内容数量以及动画过程。“`InteractionManager.runAfterInteractions接受一个唯一的回调函数作为参数。这个回调函数在导航动画完成后触发。
你的scene组件应该如下
class ExpensiveScene extends React.Component {
constructor(props, context) {
super(props, context);
this.state = {renderPlaceholderOnly: true};
}
componentDidMount() {
InteractionManager.runAfterInteractions(() => {
this.setState({renderPlaceholderOnly: false});
});
}
render() {
if (this.state.renderPlaceholderOnly) {
return this._renderPlaceholderView();
}
return (
Your full view goes here
);
}
_renderPlaceholderView() {
return (
Loading...
);
}
};
在这,你可以不局限于加载器,也可以是读取部分的类容。比如,当你加载Facebook appp时,你会看到新闻评论的占位符是一个灰色的矩形.
我们可以通过以下的几个方面,改善部分性能
initialListSize
这个属性用来指定我们第一次渲染时,要读取的行数。如果我们想尽可能的快,我们可以设置它为1, 然后可以在后续的帧中,填弃其它的行。每一次读取的行数,由pageSize决定.
**pageSize**pageSize
在使用了initialListSize之后,ListView根据pageSize来决定每一帧读取的行数,默认值为1, 但如果你的的views 非常的小,并且读取时占的资源很少, 你可以调整这个值,在找到适合你的值。
scrollRenderAheadDistance
“以像素为单位,如何预读取要加载的行?”
如果我们的列表有2000个项,而让它一次性读取,它会导致内存和计算资源的耗尽。所以scrollRenderAhead distance可以指定,超出当前视口多省,继续宣染。
removeClippedSubviews
“当它设置为true时,当本地端的superview为offscreen时 ,不在屏幕上显示的子视图offscreen(它的overflow的值为hidden) 会被删除。它可以改善长列表的滚动的性能,默认值为true.
这对于大的ListViews来说是一个非常重要。在Android, overflow的值通常为hidden. 所以我们并不需要担心它的设置,但是对于iOS来说,你需要设置row container的样式为overflow: hidden
我的组件读取时非常慢,并且不需要立即读取所有的内容
对于这个问题,我们首先可能不会想到使用ListView. 但使用ListView得当,往往是实现移定性能的关键,如我们上面讨论的,ListView向我们提供了不同的工具,可以让你视图分隔为多个帧中读取,满足特定的需求。请记住ListView也可以是水平读取。
如果你使用了ListView, 你必须提供一个rowHasChanged函数,它可以确定是否需要重新渲染一行。如果你使用的是不变的数据结构,它可以跟引用类型的相等比较一样简单。
相似的,你也可以实现shouldComponentUpdate,以确定需要重新渲染时的条件。如果你写了一个纯组件(render函数完成依赖props和state), 你可以利用PureRenderMixin来做这个事情,再一次,不可变的数据结构非常有用,可以保持它的快速性,如果你要深入的比较一个大的对像列表,使得重新渲染整个组件会更快,而且只要更少的代码.
“慢的导航转场”是这个问题的最常见的表现,但也有其它的情形。使用InteractionManager是一个最好的方法。但如果在动画期间,有大量的延迟类的工作,则可以考虑LayoutAnimation.
Animated api当前计算每一帧都是基于Javascript线程,而LayoutAnimation利用了核心动画,不会受到JS线程和主线程丢帧的影响。
常见的一种情况是弹出对话框的动画(从上向下滑动,并且淡入的动画), 而初始化和接收多个网络请求的响应,渲染对话框的的内容,并且当打开对话框时,更新视图。可以查看Animations 指南了解更多使用LayoutAnimation的信息.
警告 - LayoutAnimtion仅适用于fire-and-forget动画(“静态”动画) - 如果动画需要被中断,则你需要使用Animated.
这种情况常见于,带透明背景的文本,在一个图片之上。或者它的alpha混合的情况。可以使用shouldRasterizeIOS和renderToHardwareTextureAndroid来改进性能。
但最好不要浪用它,或者你的内存会被用完,在使用这个属性时,最好监控性能和内存的使用。如果你不计划移动一个View, 则把这些属性关闭。
在iOS, 每一次你调整一个图片组件的宽和高,它都是从原始图片中re-croped and scaled。这个过程非常昂贵,特别是对于大图来说。代替的, 我们可以使用transform: [{scale}]样式属性动画的改变大小,比如轻触一个图片,然后变为全屏。
有时,如果我们在相同的帧里改变透明度和颜色,以响应触摸事情。我们可能在onPress返回之后看不到作何的响应。如果onPress里有一个setState, 它引发大量的工作,并且有一些帧被删除掉,这时就会出现这种情况。一种解决方案是,将作何的动画包装在requestAnimationFrame处理器中。
handleOnPress() {
// Always use TimerMixin with requestAnimationFrame, setTimeout and
// setInterval
this.requestAnimationFrame(() => {
this.doExpensiveAction();
});
}
使用内置的分析器,获得更多关于Javascript 线程和主线程的信息。
在iOS中,instruments是一个非常好的工具,对于Android, 则需要学会使用systrace.