与查询不同,变更 (mutations) 通常用于创建/更新/删除数据或执行服务端副作用。为此,TanStack Query 导出了 useMutation 钩子。
以下是一个向服务器添加新待办事项的变更示例:
function App() {
const mutation = useMutation(() => {
mutationFn: (newTodo) => {
return axios.post('/todos', newTodo)
},
})
return (
<div>
{mutation.isPending ? (
'添加待办中...'
) : (
<>
{mutation.isError ? (
<div>发生错误:{mutation.error.message}</div>
) : null}
{mutation.isSuccess ? <div>待办事项已添加!</div> : null}
<button
onClick={() => {
mutation.mutate({ id: new Date(), title: '洗衣服' })
}}
>
创建待办事项
</button>
</>
)}
</div>
)
}
function App() {
const mutation = useMutation(() => {
mutationFn: (newTodo) => {
return axios.post('/todos', newTodo)
},
})
return (
<div>
{mutation.isPending ? (
'添加待办中...'
) : (
<>
{mutation.isError ? (
<div>发生错误:{mutation.error.message}</div>
) : null}
{mutation.isSuccess ? <div>待办事项已添加!</div> : null}
<button
onClick={() => {
mutation.mutate({ id: new Date(), title: '洗衣服' })
}}
>
创建待办事项
</button>
</>
)}
</div>
)
}
变更在任何时刻只能处于以下状态之一:
除了这些主要状态外,根据变更状态还可获取更多信息:
在上面的示例中,您还看到可以通过调用 mutate 函数并传递单个变量或对象来向变更函数传递变量。
仅使用变量时,变更并不特殊,但当与 onSuccess 选项、Query Client 的 invalidateQueries 方法 和 Query Client 的 setQueryData 方法 结合使用时,变更将成为非常强大的工具。
重要提示:mutate 函数是一个异步函数,这意味着在 Solid 16 及更早版本 中不能直接在事件回调中使用它。如果需要在 onSubmit 中访问事件,必须将 mutate 包装在另一个函数中。这是由于 Solid 事件池机制。
// 在 Solid 16 及更早版本中,以下代码无法工作
const CreateTodo = () => {
const mutation = useMutation(() => {
mutationFn: (event) => {
event.preventDefault()
return fetch('/api', new FormData(event.target))
},
})
return <form onSubmit={mutation.mutate}>...</form>
}
// 以下代码可以正常工作
const CreateTodo = () => {
const mutation = useMutation(() => {
mutationFn: (formData) => {
return fetch('/api', formData)
},
})
const onSubmit = (event) => {
event.preventDefault()
mutation.mutate(new FormData(event.target))
}
return <form onSubmit={onSubmit}>...</form>
}
// 在 Solid 16 及更早版本中,以下代码无法工作
const CreateTodo = () => {
const mutation = useMutation(() => {
mutationFn: (event) => {
event.preventDefault()
return fetch('/api', new FormData(event.target))
},
})
return <form onSubmit={mutation.mutate}>...</form>
}
// 以下代码可以正常工作
const CreateTodo = () => {
const mutation = useMutation(() => {
mutationFn: (formData) => {
return fetch('/api', formData)
},
})
const onSubmit = (event) => {
event.preventDefault()
mutation.mutate(new FormData(event.target))
}
return <form onSubmit={onSubmit}>...</form>
}
有时需要清除变更请求的 error 或 data。为此,可以使用 reset 函数来处理:
const CreateTodo = () => {
const [title, setTitle] = useState('')
const mutation = useMutation(() => { mutationFn: createTodo })
const onCreateTodo = (e) => {
e.preventDefault()
mutation.mutate({ title })
}
return (
<form onSubmit={onCreateTodo}>
{mutation.error && (
<h5 onClick={() => mutation.reset()}>{mutation.error}</h5>
)}
<input
type="text"
value={title}
onChange={(e) => setTitle(e.target.value)}
/>
<br />
<button type="submit">创建待办事项</button>
</form>
)
}
const CreateTodo = () => {
const [title, setTitle] = useState('')
const mutation = useMutation(() => { mutationFn: createTodo })
const onCreateTodo = (e) => {
e.preventDefault()
mutation.mutate({ title })
}
return (
<form onSubmit={onCreateTodo}>
{mutation.error && (
<h5 onClick={() => mutation.reset()}>{mutation.error}</h5>
)}
<input
type="text"
value={title}
onChange={(e) => setTitle(e.target.value)}
/>
<br />
<button type="submit">创建待办事项</button>
</form>
)
}
useMutation 提供了一些辅助选项,允许在变更生命周期的任何阶段快速简单地执行副作用。这些选项对于变更后使查询失效并重新获取 甚至乐观更新 非常有用。
useMutation(() => {
mutationFn: addTodo,
onMutate: (variables) => {
// 变更即将执行!
// 可选返回一个包含数据的上下文,用于例如回滚操作
return { id: 1 }
},
onError: (error, variables, context) => {
// 发生错误!
console.log(`回滚乐观更新,ID:${context.id}`)
},
onSuccess: (data, variables, context) => {
// 成功!
},
onSettled: (data, error, variables, context) => {
// 无论错误还是成功都会执行!
},
})
useMutation(() => {
mutationFn: addTodo,
onMutate: (variables) => {
// 变更即将执行!
// 可选返回一个包含数据的上下文,用于例如回滚操作
return { id: 1 }
},
onError: (error, variables, context) => {
// 发生错误!
console.log(`回滚乐观更新,ID:${context.id}`)
},
onSuccess: (data, variables, context) => {
// 成功!
},
onSettled: (data, error, variables, context) => {
// 无论错误还是成功都会执行!
},
})
在任何回调函数中返回 Promise 时,会先等待该 Promise 完成再调用下一个回调:
useMutation(() => {
mutationFn: addTodo,
onSuccess: async () => {
console.log('我是第一个!')
},
onSettled: async () => {
console.log('我是第二个!')
},
})
useMutation(() => {
mutationFn: addTodo,
onSuccess: async () => {
console.log('我是第一个!')
},
onSettled: async () => {
console.log('我是第二个!')
},
})
您可能希望在调用 mutate 时触发额外的回调,而不仅限于 useMutation 中定义的回调。这可用于触发组件特定的副作用。为此,可以在变更变量之后向 mutate 函数提供相同的回调选项。支持的选项包括:onSuccess、onError 和 onSettled。请注意,如果组件在变更完成前卸载,这些额外的回调将不会执行。
useMutation(() => {
mutationFn: addTodo,
onSuccess: (data, variables, context) => {
// 我会先触发
},
onError: (error, variables, context) => {
// 我会先触发
},
onSettled: (data, error, variables, context) => {
// 我会先触发
},
})
mutate(todo, {
onSuccess: (data, variables, context) => {
// 我会第二个触发!
},
onError: (error, variables, context) => {
// 我会第二个触发!
},
onSettled: (data, error, variables, context) => {
// 我会第二个触发!
},
})
useMutation(() => {
mutationFn: addTodo,
onSuccess: (data, variables, context) => {
// 我会先触发
},
onError: (error, variables, context) => {
// 我会先触发
},
onSettled: (data, error, variables, context) => {
// 我会先触发
},
})
mutate(todo, {
onSuccess: (data, variables, context) => {
// 我会第二个触发!
},
onError: (error, variables, context) => {
// 我会第二个触发!
},
onSettled: (data, error, variables, context) => {
// 我会第二个触发!
},
})
在处理连续变更时,onSuccess、onError 和 onSettled 回调的行为略有不同。当传递给 mutate 函数时,它们只会触发一次,并且仅在组件仍挂载时触发。这是因为每次调用 mutate 函数时,变更观察器会被移除并重新订阅。相反,useMutation 的处理程序会为每次 mutate 调用执行。
请注意,传递给 useMutation 的 mutationFn 很可能是异步的。在这种情况下,变更完成的顺序可能与 mutate 函数调用的顺序不同。
useMutation(() => {
mutationFn: addTodo,
onSuccess: (data, variables, context) => {
// 会被调用 3 次
},
})
const todos = ['待办 1', '待办 2', '待办 3']
todos.forEach((todo) => {
mutate(todo, {
onSuccess: (data, variables, context) => {
// 只会执行一次,针对最后一次变更(待办 3),
// 无论哪次变更先完成
},
})
})
useMutation(() => {
mutationFn: addTodo,
onSuccess: (data, variables, context) => {
// 会被调用 3 次
},
})
const todos = ['待办 1', '待办 2', '待办 3']
todos.forEach((todo) => {
mutate(todo, {
onSuccess: (data, variables, context) => {
// 只会执行一次,针对最后一次变更(待办 3),
// 无论哪次变更先完成
},
})
})
使用 mutateAsync 替代 mutate 可以获取一个 Promise,该 Promise 在成功时解析或在出错时抛出异常。例如,这可用于组合副作用。
const mutation = useMutation(() => { mutationFn: addTodo })
try {
const todo = await mutation.mutateAsync(todo)
console.log(todo)
} catch (error) {
console.error(error)
} finally {
console.log('完成')
}
const mutation = useMutation(() => { mutationFn: addTodo })
try {
const todo = await mutation.mutateAsync(todo)
console.log(todo)
} catch (error) {
console.error(error)
} finally {
console.log('完成')
}
默认情况下,TanStack Query 不会在出错时重试变更,但可以通过 retry 选项启用:
const mutation = useMutation(() => {
mutationFn: addTodo,
retry: 3,
})
const mutation = useMutation(() => {
mutationFn: addTodo,
retry: 3,
})
如果因设备离线导致变更失败,它们将在设备重新连接时按相同顺序重试。
如果需要,可以将变更持久化到存储中,并在稍后恢复。这可以通过 hydration 函数实现:
const queryClient = new QueryClient()
// 定义 "addTodo" 变更
queryClient.setMutationDefaults(['addTodo'], {
mutationFn: addTodo,
onMutate: async (variables) => {
// 取消当前待办列表的查询
await queryClient.cancelQueries({ queryKey: ['todos'] })
// 创建乐观待办事项
const optimisticTodo = { id: uuid(), title: variables.title }
// 将乐观待办事项添加到待办列表
queryClient.setQueryData(['todos'], (old) => [...old, optimisticTodo])
// 返回包含乐观待办事项的上下文
return { optimisticTodo }
},
onSuccess: (result, variables, context) => {
// 用结果替换待办列表中的乐观待办事项
queryClient.setQueryData(['todos'], (old) =>
old.map((todo) =>
todo.id === context.optimisticTodo.id ? result : todo,
),
)
},
onError: (error, variables, context) => {
// 从待办列表中移除乐观待办事项
queryClient.setQueryData(['todos'], (old) =>
old.filter((todo) => todo.id !== context.optimisticTodo.id),
)
},
retry: 3,
})
// 在某个组件中启动变更:
const mutation = useMutation(() => { mutationKey: ['addTodo'] })
mutation.mutate({ title: '标题' })
// 如果因设备离线等原因暂停了变更,
// 可以在应用退出时将暂停的变更脱水:
const state = dehydrate(queryClient)
// 然后在应用启动时重新水合:
hydrate(queryClient, state)
// 恢复暂停的变更:
queryClient.resumePausedMutations()
const queryClient = new QueryClient()
// 定义 "addTodo" 变更
queryClient.setMutationDefaults(['addTodo'], {
mutationFn: addTodo,
onMutate: async (variables) => {
// 取消当前待办列表的查询
await queryClient.cancelQueries({ queryKey: ['todos'] })
// 创建乐观待办事项
const optimisticTodo = { id: uuid(), title: variables.title }
// 将乐观待办事项添加到待办列表
queryClient.setQueryData(['todos'], (old) => [...old, optimisticTodo])
// 返回包含乐观待办事项的上下文
return { optimisticTodo }
},
onSuccess: (result, variables, context) => {
// 用结果替换待办列表中的乐观待办事项
queryClient.setQueryData(['todos'], (old) =>
old.map((todo) =>
todo.id === context.optimisticTodo.id ? result : todo,
),
)
},
onError: (error, variables, context) => {
// 从待办列表中移除乐观待办事项
queryClient.setQueryData(['todos'], (old) =>
old.filter((todo) => todo.id !== context.optimisticTodo.id),
)
},
retry: 3,
})
// 在某个组件中启动变更:
const mutation = useMutation(() => { mutationKey: ['addTodo'] })
mutation.mutate({ title: '标题' })
// 如果因设备离线等原因暂停了变更,
// 可以在应用退出时将暂停的变更脱水:
const state = dehydrate(queryClient)
// 然后在应用启动时重新水合:
hydrate(queryClient, state)
// 恢复暂停的变更:
queryClient.resumePausedMutations()
如果使用 persistQueryClient 插件 持久化离线变更,除非提供默认的变更函数,否则在页面重新加载时无法恢复变更。
这是一个技术限制。当持久化到外部存储时,只有变更的状态会被持久化,因为函数无法序列化。水合后,触发变更的组件可能未挂载,因此调用 resumePausedMutations 可能会导致错误:未找到 mutationFn。
const persister = createSyncStoragePersister({
storage: window.localStorage,
})
const queryClient = new QueryClient({
defaultOptions: {
queries: {
gcTime: 1000 * 60 * 60 * 24, // 24 小时
},
},
})
// 需要一个默认的变更函数,以便页面重新加载后可以恢复暂停的变更
queryClient.setMutationDefaults(['todos'], {
mutationFn: ({ id, data }) => {
return api.updateTodo(id, data)
},
})
export default function App() {
return (
<PersistQueryClientProvider
client={queryClient}
persistOptions={{ persister }}
onSuccess={() => {
// 从 localStorage 初始恢复成功后恢复变更
queryClient.resumePausedMutations()
}}
>
<RestOfTheApp />
</PersistQueryClientProvider>
)
}
const persister = createSyncStoragePersister({
storage: window.localStorage,
})
const queryClient = new QueryClient({
defaultOptions: {
queries: {
gcTime: 1000 * 60 * 60 * 24, // 24 小时
},
},
})
// 需要一个默认的变更函数,以便页面重新加载后可以恢复暂停的变更
queryClient.setMutationDefaults(['todos'], {
mutationFn: ({ id, data }) => {
return api.updateTodo(id, data)
},
})
export default function App() {
return (
<PersistQueryClientProvider
client={queryClient}
persistOptions={{ persister }}
onSuccess={() => {
// 从 localStorage 初始恢复成功后恢复变更
queryClient.resumePausedMutations()
}}
>
<RestOfTheApp />
</PersistQueryClientProvider>
)
}
我们还提供了一个全面的离线示例,涵盖了查询和变更。
默认情况下,所有变更并行运行——即使多次调用同一变更的 .mutate()。可以通过为变更指定带有 id 的 scope 来避免这种情况。具有相同 scope.id 的所有变更将串行运行,这意味着当它们被触发时,如果该作用域已有变更在进行中,它们将以 isPaused: true 状态开始。它们会被放入队列,并在轮到它们时自动恢复。
const mutation = useMutation(() => {
mutationFn: addTodo,
scope: {
id: 'todo',
},
})
const mutation = useMutation(() => {
mutationFn: addTodo,
scope: {
id: 'todo',
},
})
有关变更的更多信息,请查看社区资源中的 #12: 掌握 Solid Query 中的变更。