I'm proud to announce that this article is now part of the official docs.
(Before going on, check out my newest React/React Native open source project!)
React Native's FlatList
is great! But when it comes to big lists it has some flaws. There are a lot of issues and blog posts talking about how you can improve it. Here, however, as someone really concerned about this, i'll make an attempt to create an exhaustive, comprehensive and collaborative documentation about this.
I hate to be corny, but there is no silver bullet (or "magic combination") for this issue. You have to consider the trade offs of every approach and what you think is a good experience for your audience and the best strategy for your app. But fortunatelly there are several tweaks you can try and improve your FlatList
.
PR's are really needed for fixing my English, fixing wrong concepts, adding relevant information and techniques, and adding links. Giving a β is also very helpful (:
There are a lot of terms used (on docs or some issues) that were confusing for me at first. So, let's get this out of the way from the start.
-
VirtualizedList is the component behind
FlatList
, and is React Native's implementation of the 'virtual list' concept. -
Performance, in this context, imply a smooth (not chopy) scroll (and navigation in or out of your list) experience.
-
Memory consumption, in this context, is how much information about your list is being stored in memory, which could lead to a app crash.
-
Responsiveness, in this context, is your apps's ability to respond to interactions. Low responsiveness, for instance, is when you touch on a component and it waits a bit to respond, instead of responding immediately as expected.
-
Blank areas means that the VirtualizedList couldn't render your items fast enough, so you enter on a part of your list with non rendered components.
-
Window here is not your viewport but, rather, the area size in which items should be rendered.
One way to improve your FlatList
is by tweaking it's props. Here are a list of props that can help your with that.
You can set the removeClippedSubviews
prop to true, which unmount components that are off of the window.
Win: This is very memory friendly, as you will always have a little amount of rendered items instead of the whole list.
Trade offs: Be aware that this implementation can have bugs, such as missing content (mainly observed on iOS) if you use it on a component that will not unmount (such as a root navigation scene). It also can be less performant, having choppy scroll animations for big lists with complex items on not-so-good devices, as it makes crazy amounts of calculations per scroll.
You can set the maxToRenderPerBatch={number}
, which is a VirtualizedList
prop that can be passed directly to FlatList
. With this, you can control the amount of items rendered per batch, which is the next chunk of items rendered on every scroll.
Win: Setting a bigger number means less visual blank areas when scrolling (a better the fill rate).
Trade offs: More items per batch means less javascript performance, which means less responsiveness (clicking a item and opening the detail). If you have a static and non-interactive list, this could be the way to go.
While maxToRenderPerBatch
tells the amount of items rendered per batch, setting updateCellsBatchingPeriod={number}
tells to your VirtualizedList the delay, in milliseconds, between batch renders. In other words, it defines how frequently your component will be rendering the windowed items.
Win: Combining this prop with maxToRenderPerBatch
gives you the power to, for example, render more items in a less frequent batch, or less items in a more frequent batch. Which works the best for your use case.
Trade offs: Less frequent batches may cause blank areas. More frequent batches may cause responsiveness and performance loss.
You can set the initialNumToRender={number}
. This means the initial amount of items to render.
Win: You can set this value to the precise number of items that would cover the screen for every device. This can be a big performance boost when rendering the list component.
Trade offs: You are most likely to see blank areas when setting a low initialNumToRender
.
You can set the windowSize={number}
. The number passed here is a measurement unit where 1 is equivalent to your viewport height. The default value is 21, being 10 viewports above, 10 below, and one in between.
Win: If you're worried mainly about performance, you can set a bigger windowSize
so your list will run smoothly and with less blank space. If you're mainly worried about memory consumption, you can set a lower windowSize
so your rendered list will be smaller.
Trade offs: For a bigger windowSize
, you will have a bigger memory consumption. For a lower windowSize
, you will have lower performance and bigger chance of seeing blank areas.
This prop, when true, make your FlatList
rely on the older ListView
, instead of VirtualizedList
.
Win: This will make your list definitely perform better, as it removes virtualization and render all your items at once.
Trade offs: Your memory consumption goes to the roof and chances are good that a big list (100+) with complex items will crash your app. It also fires a warning that the above tweaks will not work, because you're now on a ListView.
You will see people advocation the use of this prop on some issues. But this is deprecated now. This used to do something similar to legacyImplementation
.
There are also some win-win strategies that involves your list item components. They are being managed by VirtualizedList a lot, so they need to be fast.
The more complex your components are, the slower they will render. Try to avoid a lot of logic and nesting in your list items. If you are reusing this list item component a lot in your app, create a duplicate just for your big lists and make them with less logic as possible and less nested as possible.
The heavier your components are, the slower they render. Avoid heavy images (use a cropped version for list items, as small as possible). Talk to your design team, use as little effects and interactions and information as possible in your list. Save them to your item's detail.
Implement update verification to your components. React's PureComponent implement a shouldComponentUpdate
with shallow compasion. This is expensive here, because it need to check all your props. If you want a good bit-level performance, create the strictest rules for your list item components, checking only props that could potentially change. If your list is simple enough, you could even use
shouldComponentUpdate() {
return false
}
I personally use react-native-fast-image from @DylanVann. Every image in your list is a new Image()
instance. The faster it reaches the loaded
hook, the faster your Javascript thread will be free again.
You can set the getItemLayout
to your FlatList
component. If all your list item components have the same height (or width, for a horizontal list), passing this prop removes the need for your FlatList
to dynamically calculate it every time. This is a very desirable optimization technique and if your components have dynamic size, and you really need performance, consider asking your design team if they may think of a redesign in order to perform better.
Your method should look like this, for items with height of, say, 70
:
getItemLayout = (data, index) => ({
length: 70,
offset: 70 * index,
index
})
You can set the keyExtractor
to your FlatList
component. This prop is used for caching and as the React key
to track item re-ordering. For example, if you're using your item id as the key:
keyExtractor={item => item.id}
There is a lot of discussion going on about this topic, so below I try to list the most relevant threads
The main thread about the topic
Very well documented memory leak
Optimizing list render performance in React Native (by @shichongrui)
React Native 100+ items flatlist very slow performance (on StackOverflow)