当前位置:首页 > 科技  > 软件

探索 useSyncExternalStore,一个鲜为人知的 React Hook

来源: 责编: 时间:2024-03-19 09:25:47 248观看
导读原文链接:https://blog.logrocket.com/exploring-usesyncexternalstore-react-hook/原文作者:Abhinav Anshul 翻译:一川您可能已经熟悉 React 提供的一组内置 Hook,例如 useState、useEffect、useMemo 等等。其中包括 use

原文链接:https://blog.logrocket.com/exploring-usesyncexternalstore-react-hook/f6K28资讯网——每日最新资讯28at.com

原文作者:Abhinav Anshul 翻译:一川f6K28资讯网——每日最新资讯28at.com

您可能已经熟悉 React 提供的一组内置 Hook,例如 useState、useEffect、useMemo 等等。其中包括 useSyncExternalStore Hook,它在库作者中非常常用,但在客户端 React 项目中很少见到。f6K28资讯网——每日最新资讯28at.com

在本文中,将探讨 useSyncExternalStore Hook,以更好地了解它是什么、它如何工作、为什么有用以及何时应该在前端项目中利用它。f6K28资讯网——每日最新资讯28at.com

useSyncExternalStore 简介

如果您想订阅外部数据存储,useSyncExternalStore 可能是完美的 API。大多数时候,开发人员选择 useEffect Hook。但是,如果您的数据存在于 React 树之外,则 useSyncExternalStore 可能更合适。f6K28资讯网——每日最新资讯28at.com

基本的 useSyncExternalStore API 接受三个参数:f6K28资讯网——每日最新资讯28at.com

useSyncExternalStore(subscribe, getSnapshot, getServerSnapshot)
  • subscribe 是一个回调,它接收订阅外部存储数据的函数
  • getSnapshot 是一个返回外部存储数据当前快照的函数
  • getServerSnapshot 是一个可选参数,用于发送初始存储数据的快照。可以在服务器数据的初始水合作用期间使用它

useSyncExternalStore 返回订阅的外部数据的当前快照。f6K28资讯网——每日最新资讯28at.com

考虑这样一种情况,外部数据不在React 树中——换句话说,它存在于你的前端代码或一般应用程序之外。在这种情况下,可以使用 useSyncExternalStore 订阅该数据存储。f6K28资讯网——每日最新资讯28at.com

为了更好地理解 useSyncExternalStore Hook,让我们看一个非常简单的实现。可以将其分配给一个变量(例如下面示例中的 list ),并根据需要将其呈现给 UI:f6K28资讯网——每日最新资讯28at.com

import { useSyncExternalStore } from 'react';import externalStore from './externalStore.js';function Home() {const list = useSyncExternalStore(externalStore.subscribe, externalStore.getSnapshot);  return (    <>      <section>        {list.map((itm, index) => (          <div key={index}>            <div>{itm?.title}</div>          </div>        ))}      </section>    </>  );}

如您所见, externalStore 现已订阅,将获得对 externalStore 数据执行的任何更改的实时快照。您可以使用 list 进一步映射外部源中的项目并进行实时 UI 渲染。f6K28资讯网——每日最新资讯28at.com

外部存储中的任何更改都会立即反映出来,React 将根据快照更改重新渲染 UI。f6K28资讯网——每日最新资讯28at.com

useSyncExternalStore 的用例

useSyncExternalStore Hook 是许多利基用例的理想解决方案,例如:f6K28资讯网——每日最新资讯28at.com

  • 缓存来自外部 API 的数据:由于此 Hook 主要用于订阅外部第三方数据源,因此缓存数据也变得更简单。可以使应用程序的数据与外部数据源保持同步,以后还可以将其用于离线支持
  • WebSocket 连接:由于 WebSocket 是一个“持续”连接,因此可以使用此 Hook 来实时管理 WebSocket 连接状态数据
  • 管理浏览器存储:在这种情况下,需要在 Web 浏览器存储(例如 IndexedDB 或 localStorage )与应用程序状态之间同步数据,可以使用 useSyncExternalStore 订阅更新外部store

在很多这样的情况下,这个 Hook 可能比流行的 useEffect Hook 非常有用并且更容易管理。让我们在下一节中比较这两个 Hook。f6K28资讯网——每日最新资讯28at.com

useSyncExternalStore 与 useEffect

您可以选择更常用的 useEffect Hook 来实现与上面示例类似的功能:f6K28资讯网——每日最新资讯28at.com

const [list, setList] = useState([]);  useEffect(() => {    const fetchData = async () => {      try {        // assuming externalStore has a fetchData method or it is an async operation        const newList = await externalStore.fetchData();        setList(newList);      } catch (error) {        console.error(error);      }    };    // calling the async function here    fetchData();  }, []);

但是, useEffect Hook 不提供每次状态更新的当前快照,并且它比 useSyncExternalStore Hook 更容易出错。此外,它还存在臭名昭著的重新渲染问题。接下来我们简单回顾一下这个问题。f6K28资讯网——每日最新资讯28at.com

在处理 useEffect Hook 时,可能会遇到的一个主要问题是渲染顺序。浏览器完成绘制后,只有 useEffect Hook 会触发。这种延迟(尽管是故意的)会在管理正确的事件链方面带来意想不到的错误和挑战。f6K28资讯网——每日最新资讯28at.com

考虑以下示例:f6K28资讯网——每日最新资讯28at.com

function Counter() {  const [count, setCount] = useState(0);  useEffect(() => {    console.log('count- ', count);    // Imagine some asynchronous task here, like fetching data from an API    // This could introduce a delay between the state update and the effect running    // afterwards.  }, [count]);  const increment = () => {    setCount(count + 1);  };  console.log('outside the effect count - ', count);  return (    <div>      <div>Counter</div>      <div>Count: {count}</div>      <button onClick={increment}>Increment</button>    </div>  );}export default Counter;

您可能期望计数器应用程序以简单的方式运行,其中状态更新,组件重新渲染,最后运行效果。然而,由于 API 调用的延迟,事情变得有点棘手,并且事件的顺序可能不是我们所期望的。f6K28资讯网——每日最新资讯28at.com

现在考虑一个具有许多此类副作用和不同依赖项数组的应用程序。在这种情况下,以正确的顺序跟踪状态更新将是一场噩梦。f6K28资讯网——每日最新资讯28at.com

如果您的数据位于外部并且不依赖于现有的 React API 来处理,那么您可以避免所有这些并使用 useSyncExternalStore Hook 来修复此性能差距。与 useEffect Hook 不同,此 Hook 会立即触发,不会造成任何延迟。f6K28资讯网——每日最新资讯28at.com

useSyncExternalStore 还可以防止前面提到的重新渲染问题,只要状态发生变化,您就可能会遇到 useEffect 问题。有趣的是,使用 useSyncExternalStore 订阅的状态不会重新渲染两次,从而解决了巨大的性能问题。f6K28资讯网——每日最新资讯28at.com

useSyncExternalStore 与 useState

使用 useSyncExternalStore Hook 时,您可能会觉得您只是订阅一个状态并将其分配给一个变量,类似于使用 useState Hook。然而, useSyncExternalStore 不仅仅是简单地分配状态。f6K28资讯网——每日最新资讯28at.com

useState Hook 的一个限制是它被设计为以“每个组件”的方式管理状态。换句话说,你定义的状态仅限于它自己的React组件,无法全局访问。您可以使用回调、全局强制状态,甚至可以在组件中使用 prop-drilling 状态,但这可能会减慢您的 React 应用程序的速度。f6K28资讯网——每日最新资讯28at.com

useSyncExternalStore Hook 通过设置一个全局状态来防止此问题,您可以从任何 React 组件订阅该状态,无论其嵌套有多深。更好的是,如果您正在处理非 React 代码库,您只需关心订阅事件。f6K28资讯网——每日最新资讯28at.com

useSyncExternalStore 将向您发送可以在任何 React 组件中使用的全局存储当前状态的正确快照。f6K28资讯网——每日最新资讯28at.com

使用 useSyncExternalStore 构建待办事项应用

让我们通过构建一个演示待办事项应用程序来看看 useSyncExternalStore Hook 在实际项目中有多有用。首先,创建一个用作外部全局状态的 store.js 文件。稍后我们将为我们的待办事项订阅此状态:f6K28资讯网——每日最新资讯28at.com

let todos = [];let subscribers = new Set();const store = {  getTodos() {    // getting all todos    return todos;  }, // subscribe and unsubscribe from the store using callback  subscribe(callback) {    subscribers.add(callback);    return () => subscribers.delete(callback);  },// adding todo to the state  addTodo(text) {    todos = [      ...todos,      {        id: new Date().getTime(),        text: text,        completed: false,      },    ];    subscribers.forEach((callback) => {      callback();    });  },// toggle for todo completion using id  toggleTodo(id) {    todos = todos.map((todo) => {      return todo.id === id ? { ...todo, completed: !todo.completed } : todo;    });    subscribers.forEach((callback) => callback());  },};// exporting the default store stateexport default store;

您的 store 现在已准备好在 React 组件中订阅。继续创建一个简单的 Todo 组件,该组件将通过订阅您之前创建的商店将待办事项呈现到 UI:f6K28资讯网——每日最新资讯28at.com

import { useSyncExternalStore } from "react";import store from "./store.js";function Todo() {// subscribing to the store  const todosStore = useSyncExternalStore(store.subscribe, store.getTodos);  return (    <div>      {todosStore.map((todo, index) => (        <div key={index}>           <input              type="checkbox"              value={todo.completed}              onClick={() => store.toggleTodo(todo.id)}            />            // toggle based on completion logic             {todo.completed ? <div>{todo.text}</div> : todo.text}        </div>      ))}    </div>  );}export default Todo;

至此,我们使用 useSyncExternalStore 的迷你演示项目就完成了。结果应如下所示:f6K28资讯网——每日最新资讯28at.com

图片图片f6K28资讯网——每日最新资讯28at.com

总结

React 提供了很多内置的 Hook,其中一些在开发人员中非常常用。然而,像 useSyncExternalStore 这样真正有用的 Hook 经常会被掩盖。f6K28资讯网——每日最新资讯28at.com

在这篇文章中,您已经了解了此 Hook 有许多出色的使用示例,它们不仅可以改善整体应用程序体验,还可以防止您在使用 useEffect Hook 时可能遇到的讨厌的错误。f6K28资讯网——每日最新资讯28at.com

如果您是 JavaScript 库作者,可能已经在使用它来获得使用 useEffect Hook 或 useState Hook 无法实现的性能提升。正如我们在本文中探讨的那样,如果正确使用,useSyncExternalStore Hook 可以为您节省大量的开发时间。f6K28资讯网——每日最新资讯28at.com

本文链接:http://www.28at.com/showinfo-26-77690-0.html探索 useSyncExternalStore,一个鲜为人知的 React Hook

声明:本网页内容旨在传播知识,若有侵权等问题请及时与本网联系,我们将在第一时间删除处理。邮件:2376512515@qq.com

上一篇: 学会使用aiofiles模块,让Python文件操作更高效!

下一篇: 手把手教你开发自己的VSCode插件

标签:
  • 热门焦点
  • K60 Pro官方停产 第三方瞬间涨价

    虽然没有官方宣布,但Redmi的一些高管也已经透露了,Redmi K60 Pro已经停产且不会补货,这一切都是为了即将到来的K60 Ultra铺路,属于厂家的正常操作。但有意思的是该机在停产之后
  • 小米降噪蓝牙耳机Necklace分享:听一首歌 读懂一个故事

    在今天下午的小米Civi 2新品发布会上,小米还带来了一款新的降噪蓝牙耳机Necklace,我们也在发布结束的第一时间给大家带来这款耳机的简单分享。现在大家能见到最多的蓝牙耳机
  • JavaScript 混淆及反混淆代码工具

    介绍在我们开始学习反混淆之前,我们首先要了解一下代码混淆。如果不了解代码是如何混淆的,我们可能无法成功对代码进行反混淆,尤其是使用自定义混淆器对其进行混淆时。什么是混
  • 2023 年的 Node.js 生态系统

    随着技术的不断演进和创新,Node.js 在 2023 年达到了一个新的高度。Node.js 拥有一个庞大的生态系统,可以帮助开发人员更快地实现复杂的应用。本文就来看看 Node.js 最新的生
  • 自律,给不了Keep自由!

    来源 | 互联网品牌官作者 | 李大为编排 | 又耳 审核 | 谷晓辉自律能不能给用户自由暂时不好说,但大概率不能给Keep自由。近日,全球最大的在线健身平台Keep正式登陆港交所,努力
  • 携众多高端产品亮相ChinaJoy,小米带来一场科技与人文的视听盛宴

    7月28日,全球数字娱乐领域最具知名度与影响力的年度盛会中国国际数码互动娱乐展览会(简称ChinaJoy)在上海新国际博览中心盛大开幕。作为全球领先的科
  • 超级标准版旗舰!iQOO 11S全球首发iQOO超算独显芯片

    上半年已接近尾声,截至目前各大品牌旗下的顶级旗舰都已悉数亮相,而下半年即将推出的顶级旗舰已经成为了数码圈爆料的主流,其中就包括全新的iQOO 11S系
  • 三翼鸟智能家居亮相电博会,让用户体验更真实

    2021电博会在青岛国际会展中心开幕中,三翼鸟直接把“家”搬到了现场,成为了展会的一大看点。这也是三翼鸟继9月9日发布了行业首个一站式定制智慧家平台后的
  • 荣耀Magic4 至臻版 首创智慧隐私通话 强劲影音系统

    2022年第一季度临近尾声,在该季度内,许多品牌陆续发布自己的最新产品,让大家从全新的角度来了解当今的手机技术。手机是电子设备中,更新迭代十分迅速的一款产品,基
Top