深入理解 React 的 useSyncExternalStore Hook
深入理解 React 的 useSyncExternalStore Hook
大家好,今天我们来聊聊 React 18 引入的一个新 Hook:useSyncExternalStore。这个 Hook 主要用于与外部存储同步状态,特别是在需要确保状态一致性的场景下非常有用。本文将深入探讨这个 Hook 的使用场景、工作原理,并通过代码示例来帮助大家更好地理解。
为什么需要 useSyncExternalStore?
在 React 18 之前,我们通常使用 useEffect 或者 useLayoutEffect 来订阅外部存储的变化。然而,这些方法有时会导致状态不一致的问题,特别是在并发模式下。useSyncExternalStore 旨在解决这些问题,确保状态在任何时候都是一致的。
基本用法
首先,我们来看一下 useSyncExternalStore 的基本用法。这个 Hook 接受三个参数:
subscribe: 一个函数,用于订阅外部存储的变化。getSnapshot: 一个函数,用于获取当前的存储快照。getServerSnapshot: 一个函数,用于在服务器端渲染时获取存储快照(可选)。
import React, { useSyncExternalStore } from "react";
// 模拟一个外部存储
const store = {
state: 0,
listeners: new Set(),
subscribe(listener) {
this.listeners.add(listener);
return () => this.listeners.delete(listener);
},
setState(newState) {
this.state = newState;
this.listeners.forEach((listener) => listener());
},
getState() {
return this.state;
},
};
function useStore() {
return useSyncExternalStore(
store.subscribe.bind(store),
store.getState.bind(store)
);
}
function Counter() {
const state = useStore();
return (
<div>
<p>Count: {state}</p>
<button onClick={() => store.setState(state + 1)}>Increment</button>
</div>
);
}
export default Counter;
在这个示例中,我们创建了一个简单的计数器应用。store 是一个模拟的外部存储,包含状态和订阅逻辑。useStore 是一个自定义 Hook,使用 useSyncExternalStore 来订阅存储的变化并获取当前状态。
深入理解
useSyncExternalStore 的核心在于它如何确保状态一致性。它通过同步的方式获取存储快照,避免了异步更新带来的潜在问题。在并发模式下,React 可能会在渲染过程中多次调用 getSnapshot,以确保状态的一致性。
此外,useSyncExternalStore 还支持服务器端渲染。通过传递 getServerSnapshot 参数,我们可以在服务器端获取存储快照,从而避免客户端和服务器端渲染不一致的问题。
实际应用场景
useSyncExternalStore 非常适合用于以下场景:
- 全局状态管理:例如 Redux 或者 MobX,可以使用
useSyncExternalStore来订阅全局状态的变化。 - 外部数据源:例如 WebSocket 或者其他实时数据源,可以使用这个 Hook 来确保数据的一致性。
- 复杂组件通信:在一些复杂的组件通信场景下,使用
useSyncExternalStore可以简化状态管理逻辑。
代码示例:与 Redux 集成
接下来,我们来看一个与 Redux 集成的示例。假设我们有一个简单的 Redux store:
import { createStore } from "redux";
import { Provider, useSelector, useDispatch } from "react-redux";
import React, { useSyncExternalStore } from "react";
const initialState = { count: 0 };
function reducer(state = initialState, action) {
switch (action.type) {
case "INCREMENT":
return { count: state.count + 1 };
default:
return state;
}
}
const store = createStore(reducer);
function useReduxStore() {
return useSyncExternalStore(store.subscribe, store.getState, store.getState);
}
function Counter() {
const state = useReduxStore();
const dispatch = useDispatch();
return (
<div>
<p>Count: {state.count}</p>
<button onClick={() => dispatch({ type: "INCREMENT" })}>Increment</button>
</div>
);
}
function App() {
return (
<Provider store={store}>
<Counter />
</Provider>
);
}
export default App;
在这个示例中,我们创建了一个 Redux store,并使用 useSyncExternalStore 来订阅 Redux store 的变化。这样,我们可以确保组件在任何时候都能获取到最新的状态。
总结
useSyncExternalStore 是 React 18 中一个非常强大的 Hook,特别适用于需要确保状态一致性的场景。通过本文的介绍和代码示例,希望大家能够更好地理解和应用这个 Hook。如果你在项目中遇到了状态不一致的问题,不妨试试 useSyncExternalStore。
百万大学生都在用的 AI 写论文工具,篇篇无重复 : AI 写论文
深入理解 React 的 useSyncExternalStore Hook的更多相关文章
- React+DvaJS 之 hook 路由权限控制
博客 学院 下载 GitChat TinyMind 论坛 APP 问答 商城 VIP 活动 招聘 ITeye 写博客 发Chat 登录注册 原 React+DvaJS 之 hook 路由权限控制 20 ...
- 理解 React Hooks 心智模型:必须按顺序、不能在条件语句中调用的规则
前言 自从 React 推出 hooks 的 API 后,相信大家对新 API 都很喜欢,但是它对你如何使用它会有一些奇怪的限制.比如,React 官网介绍了 Hooks 的这样一个限制: 不要在循环 ...
- 深入理解React(二) —— 数据流和事件原理
版权声明:本文由左明原创文章,转载请注明出处: 文章原文链接:https://www.qcloud.com/community/article/158 来源:腾云阁 https://www.qclou ...
- 深入理解React、Redux
深入理解React.ReduReact+Redux非常精炼,良好运用将发挥出极强劲的生产力.但最大的挑战来自于函数式编程(FP)范式.在工程化过程中,架构(顶层)设计将是一个巨大的挑战.要不然做出来的 ...
- 理解 React,但不理解 Redux,该如何通俗易懂的理解 Redux?
作者:Wang Namelos链接:https://www.zhihu.com/question/41312576/answer/90782136来源:知乎著作权归作者所有.商业转载请联系作者获得授权 ...
- 理解 React,但不理解 Redux,该如何通俗易懂的理解 Redux?(转)
作者:Wang Namelos 链接:https://www.zhihu.com/question/41312576/answer/90782136来源:知乎 解答这个问题并不困难:唯一的要求是你熟悉 ...
- 关于为什么使用React新特性Hook的一些实践与浅见
前言 关于Hook的定义官方文档是这么说的: Hook 是 React 16.8 的新增特性.它可以让你在不编写 class 的情况下使用 state 以及其他的 React 特性. 简单来说,就是在 ...
- 深入理解 React JS 中的 setState
此文主要探讨了 React JS 中的 setState 背后的机制,供深入学习 React 研究之用. 在课程 React.js入门基础与案例开发 中,有些同学会发现 React JS 中的 set ...
- 深度理解 React Suspense(附源码解析)
本文介绍与 Suspense 在三种情景下使用方法,并结合源码进行相应解析.欢迎关注个人博客. Code Spliting 在 16.6 版本之前,code-spliting 通常是由第三方库来完成的 ...
- 理解React组件的生命周期
本文作者写作的时间较早,所以里面会出现很多的旧版ES5的时代的方法.不过,虽然如此并不影响读者理解组件的生命周期.反而是作者分为几种不同的触发机制来解释生命周期的各个方法,让读者更加容易理解涉及到的概 ...
随机推荐
- java学习之旅(day.04)
运算符 算术运算符:+ ,- ,* ,/,%(取余或模运算), ++(自增),-- (自减) 赋值运算符:= 关系运算符:>, <,>=, <=, ==, !=(不等于),in ...
- 自动化部署elasticsearch三节点集群
什么是Elasticsearch? Elasticsearch 是一个开源的分布式搜索和分析引擎,构建在 Apache Lucene 的基础上.它提供了一个分布式多租户的全文搜索引擎,具有实时分析功能 ...
- win10激活方法
slmgr /ipk W269N-WFGWX-YVC9B-4J6C9-T83GX slmgr /skms zh.us.to slmgr /ato
- 【题解】A18536.星光交错的律动
题目跳转 思路:这道题可能跟博弈论有一点关系,没有学习过博弈论做起来应该问题也不大.思考一个问题,先手必胜的前提是什么? 有关更多的内容可以前往:浅谈有向无环图 先手必胜的前提是,在任何一种局面下,先 ...
- 莫烦pytorch学习记录
感谢莫烦大神Pytorch B站视频:https://www.bilibili.com/video/av15997678?p=11 一个博主的笔记:https://blog.csdn.net/Will ...
- 创建 Vue3 项目
前提条件 安装 Nodejs 下载:https://nodejs.org/en/download/prebuilt-installer 切换 npm 源 npm config set registry ...
- WNS 后台Push服务调试脚本
一.API说明 https://cloud.tencent.com/document/product/276/3212 二.推送脚本 #!/usr/local/bin/python3 # -*- ...
- mac http&git代理配置
git代理清除git config --global --unset http.proxygit config --global --unset https.proxy 输出代理:$echo $htt ...
- (C#) IIS 响应标头过滤敏感信息(如:Server/X-Powered-By等) 运维知识
背景: 再一次净网行动中,客户要求安全改造发现了接口请求的header标头中出现如图中的敏感信息. 说明: 其意义在于告知浏网站是用什么语言或者框架编写的.解决办法就是修改该响应头为一个错误的值,将攻 ...
- 剑指Offer-61.序列化二叉树(C++/Java)
题目: 请实现两个函数,分别用来序列化和反序列化二叉树 二叉树的序列化是指:把一棵二叉树按照某种遍历方式的结果以某种格式保存为字符串,从而使得内存中建立起来的二叉树可以持久保存.序列化可以基于先序 ...