Skip to content

useMemoize

Category
Export Size
242 B
Last Changed
2 weeks ago

Cache results of functions depending on arguments and keep it reactive. It can also be used for asynchronous functions and will reuse existing promises to avoid fetching the same data at the same time.

TIP

The results are not cleared automatically. Call clear() in case you no longer need the results or use own caching mechanism to avoid memory leaks.

Usage

ts
import { 
useMemoize
} from '@vueuse/core'
const
getUser
=
useMemoize
(
async (
userId
: number):
Promise
<
UserData
> =>
axios.get(`users/${
userId
}`).then(({
data
}) =>
data
),
) const
user1
= await
getUser
(1) // Request users/1
const
user2
= await
getUser
(2) // Request users/2
// ... const
user1
= await
getUser
(1) // Retrieve from cache
// ... const
user1
= await
getUser
.
load
(1) // Request users/1
// ...
getUser
.
delete
(1) // Delete cache from user 1
getUser
.
clear
() // Clear full cache
js
import { useMemoize } from '@vueuse/core'
const getUser = useMemoize(async (userId) =>
  axios.get(`users/${userId}`).then(({ data }) => data),
)
const user1 = await getUser(1) // Request users/1
const user2 = await getUser(2) // Request users/2
// ...
const user1 = await getUser(1) // Retrieve from cache
// ...
const user1 = await getUser.load(1) // Request users/1
// ...
getUser.delete(1) // Delete cache from user 1
getUser.clear() // Clear full cache

Combine with computed or asyncComputed to achieve reactivity:

ts
const 
user1
=
asyncComputed
(() =>
getUser
(1))
// ... await
getUser
.
load
(1) // Will also update user1

Resolving cache key

The key for caching is determined by the arguments given to the function and will be serialized by default with JSON.stringify. This will allow equal objects to receive the same cache key. In case you want to customize the key you can pass getKey

ts
const 
getUser
=
useMemoize
(
async (
userId
: number,
headers
:
AxiosRequestHeaders
):
Promise
<
UserData
> =>
axios.get(`users/${
userId
}`, {
headers
}).then(({
data
}) =>
data
),
{ // Use only userId to get/set cache and ignore headers
getKey
: (
userId
,
headers
) =>
userId
,
}, )
js
const getUser = useMemoize(
  async (userId, headers) =>
    axios.get(`users/${userId}`, { headers }).then(({ data }) => data),
  {
    // Use only userId to get/set cache and ignore headers
    getKey: (userId, headers) => userId,
  },
)

Customize cache mechanism

By default, the results are cached within a Map. You can implement your own mechanism by passing cache as options with following structure:

ts
export interface 
MemoizeCache
<
Key
,
Value
> {
/** * Get value for key */
get
: (
key
:
Key
) =>
Value
| undefined
/** * Set value for key */
set
: (
key
:
Key
,
value
:
Value
) => void
/** * Return flag if key exists */
has
: (
key
:
Key
) => boolean
/** * Delete value for key */
delete
: (
key
:
Key
) => void
/** * Clear cache */
clear
: () => void
}
js
export {}

Type Declarations

Show Type Declarations
ts
type 
CacheKey
= any
/** * Custom memoize cache handler */ export interface
UseMemoizeCache
<
Key
,
Value
> {
/** * Get value for key */
get
: (
key
:
Key
) =>
Value
| undefined
/** * Set value for key */
set
: (
key
:
Key
,
value
:
Value
) => void
/** * Return flag if key exists */
has
: (
key
:
Key
) => boolean
/** * Delete value for key */
delete
: (
key
:
Key
) => void
/** * Clear cache */
clear
: () => void
} /** * Memoized function */ export interface
UseMemoizeReturn
<
Result
,
Args
extends unknown[]> {
/** * Get result from cache or call memoized function */ (...
args
:
Args
):
Result
/** * Call memoized function and update cache */
load
: (...
args
:
Args
) =>
Result
/** * Delete cache of given arguments */
delete
: (...
args
:
Args
) => void
/** * Clear cache */
clear
: () => void
/** * Generate cache key for given arguments */
generateKey
: (...
args
:
Args
) =>
CacheKey
/** * Cache container */
cache
:
UseMemoizeCache
<
CacheKey
,
Result
>
} export interface
UseMemoizeOptions
<
Result
,
Args
extends unknown[]> {
getKey
?: (...
args
:
Args
) => string | number
cache
?:
UseMemoizeCache
<
CacheKey
,
Result
>
} /** * Reactive function result cache based on arguments * * @__NO_SIDE_EFFECTS__ */ export declare function
useMemoize
<
Result
,
Args
extends unknown[]>(
resolver
: (...
args
:
Args
) =>
Result
,
options
?:
UseMemoizeOptions
<
Result
,
Args
>,
):
UseMemoizeReturn
<
Result
,
Args
>

Source

SourceDocs

Contributors

Anthony Fu
SerKo
Anthony Fu
Mikhailov Nikita
Jelf
freakzlike

Changelog

v13.6.0 on
d32f8 - refactor: add @__NO_SIDE_EFFECTS__ annotations to all pure functions (#4907)
v12.0.0-beta.1 on
0a9ed - feat!: drop Vue 2 support, optimize bundles and clean up (#4349)
v10.8.0 on
a086e - fix: stricter types
v10.7.0 on
fecbe - fix: use shallowReactive to wrap Map

Released under the MIT License.

Build faster with AI
New Masterclass to help you leverage AI in your Vue workflow
Get Early Access