Cache Behavior
Last updated
Was this helpful?
Last updated
Was this helpful?
A key feature of RTK Query is its management of cached data. When data is fetched from the server, RTK Query will store the data in the Redux store as a 'cache'. When an additional request is performed for the same data, RTK Query will provide the existing cached data rather than sending an additional request to the server.
RTK Query provides a number of concepts and tools to manipulate the cache behaviour and adjust it to your needs.
With RTK Query, caching is based on:
API endpoint definitions
The serialized query parameters used when components subscribe to data from an endpoint
Active subscription reference counts
When a subscription is started, the parameters used with the endpoint are serialized and stored internally as a queryCacheKey
for the request. Any future request that produces the same queryCacheKey
(i.e. called with the same parameters, factoring serialization) will be de-duped against the original, and will share the same data and updates. i.e. two separate components performing the same request will use the same cached data.
When a request is attempted, if the data already exists in the cache, then that data is served and no new request is sent to the server. Otherwise, if the data does not exist in the cache, then a new request is sent, and the returned response is stored in the cache.
Subscriptions are reference-counted. Additional subscriptions that ask for the same endpoint+params increment the reference count. As long as there is an active 'subscription' to the data (e.g. if a component is mounted that calls a useQuery
hook for the endpoint), then the data will remain in the cache. Once the subscription is removed (e.g. when last component subscribed to the data unmounts), after an amount of time (default 60 seconds), the data will be removed from the cache. The expiration time can be configured with the keepUnusedDataFor
property for the , as well as on a basis.
Imagine an endpoint that expects an id
as the query param, and 4 components mounted which are requesting data from this same endpoint:
While four components are subscribed to the endpoint, there are only three distinct combinations of endpoint + query parameters. Query parameters 1
and 2
will each have a single subscriber, while query parameter 3
has two subscribers. RTK Query will make three distinct fetches; one for each unique set of query parameters per endpoint.
If 'ComponentThree' is unmounted in the example above, regardless of how much time passes, the data will remain in the cache due to 'ComponentFour' still being subscribed to the same data, and the subscribe reference count will be 1
. However, once 'ComponentFour' unmounts, the subscriber reference count will be 0
. The data will remain in the cache for the remainder of the expiration time. If no new subscription has been created before the timer expires, the cached data will finally be removed.
On top of the default behaviour, RTK Query provides a number of methods to re-fetch data earlier in scenarios where it should be considered invalid, or is otherwise deemed suitable to be 'refreshed'.
keepUnusedDataFor
This value can be configured using the keepUnusedDataFor
option for both the API definition, as well as per-endpoint. Note that the per-endpoint version, if provided, will overrule a setting on the API definition.
Providing a value to keepUnusedDataFor
as a number in seconds specifies how long the data should be kept in the cache after the subscriber reference count reaches zero.
refetch
/initiate
Calling the refetch
function will force refetch the associated query.
Alternatively, you can dispatch the initiate
thunk action for an endpoint, passing the option forceRefetch: true
to the thunk action creator for the same effect.
refetchOnMountOrArgChange
refetchOnMountOrArgChange
is used to encourage re-fetching in additional situations where the default behavior would instead serve te cached data.
refetchOnMountOrArgChange
accepts either a boolean value, or a number as time in seconds.
Passing true
for this property will cause the endpoint to always refetch when a new subscriber to the query is added. If passed to an individual hook call and not the api definition itself, then this applies only to that hook call. I.e., when the component calling the hook mounts, or the argument changes, it adds a new subscriber for that query, and will always refetch, regardless of whether cached data for the endpoint + arg combination already exists.
Passing a number
as a value in seconds will use the following behavior:
At the time a query subscription is created:
if there is an existing query in the cache, it will compare the current time vs the last fulfilled timestamp for that query,
It will refetch if the provided amount of time in seconds has elapsed.
If there is no query, it will fetch the data.
If there is an existing query, but the amount of time specified since the last query has not elapsed, it will serve the existing cached data.
refetchOnFocus
If you specify this option alongside skip: true
, this will not be evaluated until skip is false.
refetchOnReconnect
If you specify this option alongside skip: true
, this will not be evaluated until skip
is false.
RTK Query deliberately does not implement a cache that would deduplicate identical items across multiple requests. There are several reasons for this:
A fully normalized shared-across-queries cache is a hard problem to solve
We don't have the time, resources, or interest in trying to solve that right now
In many cases, simply re-fetching data when it's invalidated works well and is easier to understand
At a minimum, RTKQ can help solve the general use case of "fetch some data", which is a big pain point for a lot of people
As an example, say that we have an API slice with getTodos
and getTodo
endpoints, and our components make the following queries:
getTodos()
getTodos({filter: 'odd'})
getTodo({id: 1})
Each of these query results would include a Todo object that looks like {id: 1}
.
In a fully normalized de-duplicating cache, only a single copy of this Todo object would be stored. However, RTK Query saves each query result independently in the cache. So, this would result in three separate copies of this Todo being cached in the Redux store. However, if all the endpoints are consistently providing the same tags (such as {type: 'Todo', id: 1}
), then invalidating that tag will force all the matching endpoints to refetch their data for consistency.
There are a couple additional points that can help here:
Two components are mounted, each with the same endpoints query (useGetUsersQuery(2)
). You will be able to observe that when toggling off the components, the subscriber reference count will be reduced. After toggling off both components such that the subscriber reference count reaches zero, you will observe the cached data under the Queries
section will persist for 5 seconds (the value of keepUnusedDataFor
provided for the endpoint in this demo). If the subscriber reference count remains at 0 for the full duration, the cached data will then be removed from the store.
Data is kept in the cache as long is there is at least one active subscriber interested in that endpoint + parameter combination. When the subscriber reference count reaches zero, a timer is set, and if there are no new subscriptions to that data by the time the timer expires, the cached data will be removed. The default expiration is 60 seconds, which can be configured both for the , as well as on a basis.
As mentioned above under and , by default, data will remain in the cache for 60 seconds after the subscriber reference count hits zero.
In order to achieve complete granular control over re-fetching data, you can use the refetch
function returned as a result property from a or hook.
Queries can be encouraged to re-fetch more frequently than usual via the property. This can be passed to the endpoint as a whole, to individual hook calls, or when dispatching the action.
Passing false
(the default value) for this property will use the default behavior .
The option allows you to control whether RTK Query will try to refetch all subscribed queries after the application window regains focus.
Note that this requires to have been called.
This option is available on both the api definition with , as well as on the , , , and hooks.
The option on allows you to control whether RTK Query will try to refetch all subscribed queries after regaining a network connection.
Note that this requires to have been called.
This option is available on both the api definition with , as well as on the , , , and hooks.
RTK Query uses an optional system to automate re-fetching for query endpoints that have data affected by mutation endpoints.
See for full details on this concept.
The Redux docs have always recommended to enable easily finding items by ID and updating them in the store, and was designed to help manage normalized state. Those concepts are still valuable and don't go away. However, if you're using RTK Query to manage caching data, there's less need to manipulate the data that way yourself.
The generated query hooks have that allow components to read individual pieces of data from a query result. As an example, a <TodoList>
component might call useTodosQuery()
, and each individual <TodoListItem>
could use the same query hook but select from the result to get the right todo object.
You can use the to modify the fetched data so that it's , such as using createEntityAdapter
to normalize the data for this one response before it's inserted into the cache.
This example is a live demo of how the subscriber reference count and the value of keepUnusedDataFor
interact with each other. The Subscriptions
and Queries
(including the cached data) are shown in the demo for you to visualize (note that this can also be viewed in the ).