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

震惊!用 Suspense 解决请求依赖的复杂场景居然这么简单!

来源: 责编: 时间:2024-06-11 17:52:06 112观看
导读有一种复杂场景 React 新手经常处理不好。那就是一个页面有多个模块,每个模块都有自己的数据需要请求。与此同时,可能部分模块的数据还要依赖父级的异步数据才能正常请求自己的数据。如下图所示,当我们直接访问该页面时,

有一种复杂场景 React 新手经常处理不好。qpf28资讯网——每日最新资讯28at.com

那就是一个页面有多个模块,每个模块都有自己的数据需要请求。与此同时,可能部分模块的数据还要依赖父级的异步数据才能正常请求自己的数据。如下图所示,当我们直接访问该页面时,页面请求的数据就非常多。而且这些数据还有一定的先后依赖关系。qpf28资讯网——每日最新资讯28at.com

qpf28资讯网——每日最新资讯28at.com

一、重新考虑初始化

和之前的方案一样,我们先定义父组件的请求接口。qpf28资讯网——每日最新资讯28at.com

const getMessage = async () => {  const res = await fetch('https://api.chucknorris.io/jokes/random')  return res.json()}

然后在父组件中,将 getMessage() 执行之后返回的 promise 作为状态存储在 useState 中。这样,当我点击时,只需要重新执行依次 getMessage() 就可以更新整个组件。qpf28资讯网——每日最新资讯28at.com

const [  messagePromise,   setMessagePromise] = useState(null)

但是此时我们发现,messagePromise 并没有初始值,因此初始化时,接口并不会请求。这种情况下,有两种交互我们需要探讨。一种是通过点击按钮来初始化接口。另外一种就是组件首次渲染就要初始化接口。qpf28资讯网——每日最新资讯28at.com

我们之前的案例中,使用了取巧的方式,在函数组件之外提前获取了数据,这会导致访问任何页面该数据都会加载,因此并非合适的手段。qpf28资讯网——每日最新资讯28at.com

// 我们之前的案例这样做是一种取巧的方式const api = getMessage()function Message() {  ...

但是如果我们直接把 getMessage() 放在组件内部执行,也存在不小的问题。因为当组件因为其他的状态发生变化需要重新执行时,此时 getMessage() 也会冗余的多次执行。qpf28资讯网——每日最新资讯28at.com

// 此时会冗余多次执行const [  messagePromise,   setMessagePromise] = useState(getMessage())

理想的情况是 getMessage() 只在组件首次渲染时执行依次,后续状态的改变就不在执行。而不需要多次执行。qpf28资讯网——每日最新资讯28at.com

我们先来考虑通过点击事件初始化接口的交互。此时我们可以先设置 messagePrmoise 的初始值为 null。qpf28资讯网——每日最新资讯28at.com

const [  messagePromise,   setMessagePromise] = useState(null)

不过这样做有一个小问题就是如果我将 messagePromise 值为 null 时传递给了子组件。那么子组件就会报错,因此我们需要特殊处理。一种方式就是在子组件内部判断。qpf28资讯网——每日最新资讯28at.com

const MessageOutput = ({messagePromise}) => {  if (!messagePromise) return  const messageContent = use(messagePromise)

或者:qpf28资讯网——每日最新资讯28at.com

// 这种写法是在需要默认显示状态时的方案const MessageOutput = ({messagePromise}) => {  const messageContent = messagePromise ? use(messagePromise) : {value: '默认值'}

另外一种思路就是设置一个状态,子组件基于该状态的值来是否显示。然后在点击时将其设置为 true。qpf28资讯网——每日最新资讯28at.com

const [show, setShow] = useState(false)function __clickHandler() {  setMessagePromise(getMessage())  setShow(true)}
{show && <MessageContainer messagePromise={messagePromise} />}

另外一种交互思路就是初始化时就需要马上请求数据。此时我们为了确保 getMessage() 只执行一次,可以新增一个非 state 状态来记录组件的初始化情况。默认值为 false,初始化之后设置为 true。qpf28资讯网——每日最新资讯28at.com

const i = useRef(false)let __api = i.current ? null : getMessage()const [  messagePromise,   setMessagePromise] = useState(null)

然后在 useEffect 中,将其设置为 true,表示组件已经初始化过了。qpf28资讯网——每日最新资讯28at.com

useEffect(() => {  i.current = true}, [])

这是利用 useState 的内部机制,初始化值只会赋值一次来做到的。从而我们可以放心更改后续 __api 的值为 null.qpf28资讯网——每日最新资讯28at.com

从这个细节的角度来说,函数组件多次执行的确会给开发带来一些困扰,Vue3/Solid 只执行一次的机制会更舒适一些,不过处理得当也能避免这个问题。qpf28资讯网——每日最新资讯28at.com

二、Suspense 嵌套

接下来,我们需要考虑的就是 Suspense 嵌套执行的问题就行了。这个执行起来非常简单。我们只需要将有异步请求的模块用 Suspense 包裹起来当成一个子组件。然后该子组件可以当成一个常规的子组件作为 Suspense 组件的子组件。qpf28资讯网——每日最新资讯28at.com

例如,我们声明一个子组件如下所示:qpf28资讯网——每日最新资讯28at.com

const getApi = async () => {  const res = await fetch('https://api.chucknorris.io/jokes/random')  return res.json()}export default function Index(props) {  const api = getApi()  return (    <div>      <div id='tips'>多个 Suspense 嵌套,子组件第一部分</div>      <div className="content">        <div className='_05_dou1_message'>父级消息: {props.value}</div>        <Suspense fallback={<div>Loading...</div>}>          <Item api={api} />        </Suspense>      </div>    </div>  )}const Item = ({api}) => {  const joke = api ? use(api) : {value: 'nothing'}  return (    <div className='_03_a_value_update'>子级消息:{joke.value}</div>  )}

然后我可以将这个子组件放在 Suspense 内就可以了。qpf28资讯网——每日最新资讯28at.com

import DouPlus1 from './Dou1'import DouPlus2 from './Dou2'
const MessageOutput = ({messagePromise}) => {  const messageContent = use(messagePromise)  return (    <div>      <p>{messageContent.value}</p>      <DouPlus1 value={messageContent.value} />      <DouPlus2 value={messageContent.value} />    </div>  )}

在另外一个子组件中,我们还设计了内部状态,用于实现切换按钮,来增加页面交互的复杂度。并且每次切换都会请求接口。qpf28资讯网——每日最新资讯28at.com

qpf28资讯网——每日最新资讯28at.com

如果切换时,上一个接口没有请求完成,React 会自己处理好数据的先后问题。不需要我们额外考虑竞态条件的情况。完整代码如下:qpf28资讯网——每日最新资讯28at.com

var tabs = ['首页', '视频', '探索']export default function Index() {  var r = useRef(false)  var api = r.current ? null : getApi()  const [promise, setPromise] = useState(api)  const [current, setCurrent] = useState(0)  useEffect(() => {    r.current = true  }, [])  return (    <div>      <div id='tips'>多个 Suspense 嵌套,子组件第二部分</div>      <div className="content">        {tabs.map((item, index) => (          <button             id='btn_05_item'             className={current == index ? 'active' : ''}            onClick={() => {              setCurrent(index)              setPromise(getApi())            }}            key={item}          >{item}</button>        ))}                <Suspense fallback={<div className='_05_a_value_item'>Loading...</div>}>          <Item api={promise} />        </Suspense>      </div>    </div>  )}const Item = ({api}) => {  const joke = use(api)  return (    <div className='_05_a_value_item'>{joke.value}</div>  )}

三、总结

当我们要在复杂交互的情况下使用嵌套 Suspense 来解决问题,如果我们组件划分得当、与数据依赖关系处理得当,那么代码就会相当简单。不过这对于开发者来说,会有另外一个层面的要求。那就是如何合理的处理好组件归属问题。qpf28资讯网——每日最新资讯28at.com

许多前端页面开发难度往往都是由于组件划分不合理,属性归属问题处理不够到位导致的。因此 Suspense 在这个层面有了一个刚需,开发者必须要具备合理划分组件的能力,否则即使使用了 Suspense,也依然可能导致页面一团混乱。qpf28资讯网——每日最新资讯28at.com

本文链接://www.dmpip.com//www.dmpip.com/showinfo-26-93090-0.html震惊!用 Suspense 解决请求依赖的复杂场景居然这么简单!

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

上一篇: C++ 首度超越 C 语言仅次榜首 Python,TIOBE 编程指数六月排行榜公布

下一篇: PHP 服务实现性能剖析、跟踪和可观察性14444444444444】=102102102102102102102102102102102102102102102102实践

标签:
  • 热门焦点
  • 女孩租房开2小时空调用完100元电费引热议:5级能耗惹不起 月薪过万电费也交不起

    女孩租房开2小时空调用完100元电费引热议:5级能耗惹不起 月薪过万电费也交不起

    近日,江苏苏州一女孩租房当天充值了100元电费,开着空调不到2小时发现电费已用完。对于为什么这个快,房东表示,电表坏了这种情况很多,之前也遇到过,给租客换
  • 28个SpringBoot项目中常用注解,日常开发、求职面试不再懵圈

    28个SpringBoot项目中常用注解,日常开发、求职面试不再懵圈

    前言在使用SpringBoot开发中或者在求职面试中都会使用到很多注解或者问到注解相关的知识。本文主要对一些常用的注解进行了总结,同时也会举出具体例子,供大家学习和参考。注解
  • 摸鱼心法第一章——和配置文件说拜拜

    摸鱼心法第一章——和配置文件说拜拜

    为了能摸鱼我们团队做了容器化,但是带来的问题是服务配置文件很麻烦,然后大家在群里进行了“亲切友好”的沟通图片图片图片图片对比就对比,简单对比下独立配置中心和k8s作为配
  • 2023 年的 Node.js 生态系统

    2023 年的 Node.js 生态系统

    随着技术的不断演进和创新,Node.js 在 2023 年达到了一个新的高度。Node.js 拥有一个庞大的生态系统,可以帮助开发人员更快地实现复杂的应用。本文就来看看 Node.js 最新的生
  • 自动化在DevOps中的力量:简化软件开发和交付

    自动化在DevOps中的力量:简化软件开发和交付

    自动化在DevOps中扮演着重要角色,它提升了DevOps的效能。通过自动化工具和方法,DevOps团队可以实现以下目标:消除手动和重复性任务。简化流程。在整个软件开发生命周期中实现更
  • 重估百度丨“晚熟”的百度云,能等到春天吗?

    重估百度丨“晚熟”的百度云,能等到春天吗?

    &copy;自象限原创作者|程心排版|王喻可2016年7月13日,百度云计算战略发布会在北京举行,宣告着百度智能云的正式启程。彼时的会场座无虚席,甚至排队排到了门外,在场的所有人几乎都
  • 得物宠物生意「狂飙」,发力“它经济”

    得物宠物生意「狂飙」,发力“它经济”

    作者|花花小萌主近日,得物宣布正式上线宠物鉴别,通过得物App内的&ldquo;在线鉴别&rdquo;,可找到鉴别宠物的选项。通过上传自家宠物的部位细节,就能收获拥有专业资质认证的得物鉴
  • 大厂卷向扁平化

    大厂卷向扁平化

    来源:新熵作者丨南枝 编辑丨月见大厂职级不香了。俗话说,兵无常势,水无常形,互联网企业调整职级体系并不稀奇。7月13日,淘宝天猫集团启动了近年来最大的人力制度改革,目前已形成一
  • 利用职权私自解除被封帐号 Meta开除20多名员工

    利用职权私自解除被封帐号 Meta开除20多名员工

    11月18日消息,据外媒援引知情人士表示,过去一年时间内,Facebook母公司Meta解雇或处罚了20多名员工以及合同工,指控这些人通过内部系统以不当方式重置用户帐号,其
Top
Baidu
map