优化 FlatList 配置
术语
-
VirtualizedList:
FlatList背后的组件(React Native 对虚拟列表概念的实现。) -
内存消耗: 你的列表有多少信息被存储在内存中,这可能导致应用崩溃。
-
响应能力: 应用响应交互的能力。例如,响应能力低是指你触摸组件后它等待片刻才响应,而不是像预期那样立即响应。
-
空白区域: 当
VirtualizedList无法足够快地渲染你的项时,你可能会进入列表中没有渲染组件的部分,表现为空白空间。 -
视口: 渲染到像素的可见内容区域。
-
窗口: 应该挂载项的区域,通常比视口大得多。
属性
以下是一些可以帮助提高 FlatList 性能的属性:
removeClippedSubviews
| 类型 | 默认值 |
|---|---|
| Boolean | 在 Android 上为 true,否则为 false |
如果为 true,视口之外的视图会自动从原生视图层级中分离。
优点: 这减少了主线程上的耗时,从而通过排除视口之外的视图参与原生渲染和绘制遍历,降低了掉帧的风险。
缺点: 请注意,此实现可能存在 bug,例如内容丢失(主要在 iOS 上观察到),尤其是当你使用复杂的变换和/或绝对定位时。还要注意,这不会节省大量内存,因为视图没有被释放,只是分离了。
maxToRenderPerBatch
| 类型 | 默认值 |
|---|---|
| Number | 10 |
这是一个可以通过 FlatList 传递的 VirtualizedList 属性。这控制了每批渲染的项数量,即每次滚动时渲染的下一块项。
优点: 设置更大的数字意味着滚动时视觉空白区域更少(增加了填充率)。
缺点: 每批更多的项意味着更长的 JavaScript 执行时间,可能会阻塞其他事件处理,如点击,损害响应能力。
updateCellsBatchingPeriod
| 类型 | 默认值 |
|---|---|
| Number | 50 |
虽然 maxToRenderPerBatch 告诉每批渲染的项数量,但设置 updateCellsBatchingPeriod 告诉你的 VirtualizedList 批渲染之间的延迟毫秒数(你的组件渲染窗口项的频率)。
优点: 将此属性与 maxToRenderPerBatch 结合使用,使你可以,例如,在不那么频繁的批中渲染更多项,或在更频繁的批中渲染更少项。
缺点: 不那么频繁的批可能导致空白区域,更频繁的批可能导致响应能力问题。
initialNumToRender
| 类型 | 默认值 |
|---|---|
| Number | 10 |
初始渲染的项数量。
优点: 定义覆盖每个设备屏幕的确切项数。这可以大幅提升初始渲染的性能。
缺点: 设置较低的 initialNumToRender 可能导致空白区域,尤其是如果它太小而无法在初始渲染时覆盖视口。
windowSize
| 类型 | 默认值 |
|---|---|
| Number | 21 |
这里传递的数字是一个测量单位,其中 1 相当于你的视口高度。默认值是 21(上方 10 个视口,下方 10 个,中间一个)。
优点: 更大的 windowSize 将导致滚动时看到空白空间的机会更少。另一方面,更小的 windowSize 将导致同时挂载的项更少,节省内存。
缺点: 对于更大的 windowSize,你将有更多的内存消耗。对于更小的 windowSize,你将有更大的机会看到空白区域。
列表项
以下是关于列表项组件的一些提示。它们是列表的核心,所以需要快速。
使用基础组件
你的组件越复杂,渲染就越慢。尽量避免在列表项中使用大量逻辑和嵌套。如果你在应用中大量重用此列表项组件,请专门为大型列表创建一个组件,并使其逻辑和嵌套尽可能少。
使用轻量组件
你的组件越重,渲染就越慢。避免使用大图片(对列表项使用裁剪版本或缩略图,尽可能小)。与设计团队沟通,在列表中尽可能少地使用效果、交互和信息。在项的详情中展示它们。
使用 memo()
React.memo() 创建一个记忆化组件,仅当传递给组件的 props 变化时才会重新渲染。我们可以使用此函数来优化 FlatList 中的组件。
import React, {memo} from 'react';
import {View, Text} from 'react-native';
const MyListItem = memo(
({title}: {title: string}) => (
<View>
<Text>{title}</Text>
</View>
),
(prevProps, nextProps) => {
return prevProps.title === nextProps.title;
},
);
export default MyListItem;
在这个例子中,我们确定了 MyListItem 应该只在 title 变化时重新渲染。我们将比较函数作为第二个参数传递给 React.memo(),以便组件仅在指定的 prop 变化时重新渲染。如果比较函数返回 true,组件将不会重新渲染。
使用缓存优化的图片
你可以使用社区包(例如来自 Dream11 的 @d11/react-native-fast-image)来获得性能更好的图片。列表中的每个图片都是一个 new Image() 实例。它越快到达 loaded 钩子,你的 JavaScript 线程就越快再次空闲。
使用 getItemLayout
如果你所有的列表项组件都有相同的高度(对于水平列表则是宽度),提供 getItemLayout 属性可以消除 FlatList 管理异步布局计算的需要。这是一个非常理想的优化技术。
如果你的组件具有动态大小且你确实需要性能,考虑询问你的设计团队他们是否可以为了更好性能而重新设计。
使用 keyExtractor 或 key
你可以为 FlatList 组件设置 keyExtractor。此属性用于缓存并作为 React key 来跟踪项重新排序。
你也可以在项组件中使用 key 属性。
避免在 renderItem 上使用匿名函数
对于函数组件,将 renderItem 函数移到返回的 JSX 之外。此外,确保它包裹在 useCallback 钩子中,以防止每次渲染时重新创建。
对于类组件,将 renderItem 函数移到 render 函数之外,这样它就不会每次调用 render 函数时重新创建自己。
const renderItem = useCallback(({item}) => (
<View key={item.key}>
<Text>{item.title}</Text>
</View>
), []);
return (
// ...
<FlatList data={items} renderItem={renderItem} />;
// ...
);