返回顶部
首页 > 资讯 > 前端开发 > JavaScript >React useEffect不支持async function示例分析
  • 153
分享到

React useEffect不支持async function示例分析

2024-04-02 19:04:59 153人浏览 泡泡鱼
摘要

目录引言React为什么这么设计呢?简单改造1、简单改造的写法(不推荐)2、把异步提取成单独函数或自定义hook(推荐)引言 useEffect相比大家都耳熟能详啦,如下这种写法,应

引言

useEffect相比大家都耳熟能详啦,如下这种写法,应该是非常常见的需求。

useEffect(async () => {
   await getPoiInfo(); // 请求数据
}, []);

但是 React 本身并不支持这么做,理由是 effect function 应该返回一个销毁函数(effect:是指return返回的cleanup函数),如果 useEffect 第一个参数传入 async,返回值则变成了 Promise,会导致 react 在调用销毁函数的时候报错 :function.apply is undefined。

React为什么这么设计呢?

1、useEffect 的返回值是要在卸载组件时调用的,React 需要在 mount 的时候马上拿到这个值,不然就乱套了

2、useEffect() 可能有个潜在逻辑:第二次触发 useEffect 里的回调前,前一次触发的行为都执行完成,返回的清理函数也执行完成。这样逻辑才清楚。而如果是异步的,情况会变得很复杂,可能会很容易写出有 bug 的代码。

下面有两种改进的方法大家可以参考下:

简单改造

1、简单改造的写法(不推荐)

第一种 在内部创建一个异步函数anyNameFunction,等待他的结果,然后调用setData

但是这种方式存在一个问题,如果asyncFunction请求有依赖外部的参数,如果不更新requestData 的 effect 的依赖,effect 就不会同步 props 和 state 带来的变更,也就不回重新请求数据

useEffect(() => {
    // Create an scoped async function in the hook
    // 注意如果函数没有使用组件内的任何值,可以把它提到组件外面去定义
    // 下面代码可以提到外面,可以自由地在 effect 中使用,下面就不改啦
    async function asyncFunction() {
      await requestData();
      setData(data)
    }
    // Execute the created function directly
    anyNameFunction();
 }, []); // 这里设置成[]数组,因为我们只想在挂载的时候运行它一次

或者 useEffect中异步函数采用IIFE写法( Immediately Invoked Function Expression即立即调用的函数式表达式)

useEffect(() => {
  // Using an IIFE
  (async function anyNameFunction() {
    await requestData();
  })();
}, []);

2、把异步提取成单独函数或自定义hook(推荐)

第一种自定义 hook包裹,然后再effect中通过promise.then调用(GitHub上大佬给的答案:github)

// 自定义hook
function useAsyncEffect(effect: () => Promise<void | (() => void)>, dependencies?: any[]) {
  return useEffect(() => {
    const cleanupPromise = effect()
    return () => { cleanupPromise.then(cleanup => cleanup && cleanup()) }
  }, dependencies)
}
// 使用
useAsyncEffect(async () => {
    const count = await fetchData()
    setCount(count)
  }, [fetchData])

或者利用useCallback 包装成hook

useCallback 本质上是添加了一层依赖检查,使用useCallback,函数完全可以参与到数据流中,可以说如果一个函数的输入改变了,这个函数就改变了,如果没有,函数也不会改变。

下面的例子中会依赖 type ,如果 type 保持不变,requestData 也会保持不变,effect 也不会重新运行,但是如果 type 修改了,requestData 也会随之改变,因此会重新请求数据。

// 封装
const requestData = useCallback(async () => {
  changeLoading(true);
  changeError(false);
  changeList([]);
  requestapi.getFeature({ type }).then((data) => {
    if (data) {
      changeList(data);
    }
  }).catch((e) => {
    changeError(true);
  }).finally(() => {
    changeLoading(false);
  });
}, [type]); // type改变会重新生成函数
// 普通接口请求
useEffect(() => { 
  requestData();
}, [requestData]);
// 单独处理外层刷新的接口请求
// refreshing是props传递的过来的,不应该与state状态改变混在一起,这也是hook的优势,将不相关的状态逻辑拆分成更细粒度
useEffect(() => { 
  if (!refreshing) { 
    return;
  }
  requestData().then(() => {
    getRefreshStatus(false);
  });
}, [refreshing]);

关于为什么不支持异步的原理可以看下这篇文章里通过源码的分析:useEffect 中为啥不能使用 async

有任何疑问欢迎评论沟通,我会继续更新!

其他相关文档:

https://heptaluan.github.io/2020/11/07/React/17/

Https://www.robinwieruch.de/react-hooks-fetch-data/

以上就是React useEffect不支持async function示例分析的详细内容,更多关于useEffect不支持async function的资料请关注编程网其它相关文章!

--结束END--

本文标题: React useEffect不支持async function示例分析

本文链接: https://www.lsjlt.com/news/154050.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

猜你喜欢
  • React useEffect不支持async function示例分析
    目录引言React为什么这么设计呢?简单改造1、简单改造的写法(不推荐)2、把异步提取成单独函数或自定义hook(推荐)引言 useEffect相比大家都耳熟能详啦,如下这种写法,应...
    99+
    2024-04-02
  • react中useEffect闭包的示例分析
    这篇文章主要介绍react中useEffect闭包的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!问题代码看一段因为useEffect导致的闭包问题代码const btn = u...
    99+
    2023-06-15
  • ThinkJS2.1支持TypeScript的示例分析
    这篇文章主要为大家展示了“ThinkJS2.1支持TypeScript的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“ThinkJS2.1支持TypeS...
    99+
    2024-04-02
  • Spring Boot对jdbc支持的示例分析
    这篇文章将为大家详细讲解有关Spring Boot对jdbc支持的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。项目结构pom.xmlpom.xml:<xml version=&q...
    99+
    2023-06-20
  • underscore之function的示例分析
    小编给大家分享一下underscore之function的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!因为unders...
    99+
    2024-04-02
  • MySQL支不支持事务嵌套的案例分析
    MySQL支不支持事务嵌套的案例分析?这个问题可能是我们日常学习或工作经常见到的。希望通过这个问题能让你收获颇深。下面是小编给大家带来的参考内容,让我们一起来看看吧!最近开发中遇到了使用MySQL,多次开启...
    99+
    2024-04-02
  • 支持Angular2表格控件的示例分析
    这篇文章主要为大家展示了“支持Angular2表格控件的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“支持Angular2表格控件的示例分析”这篇文章吧...
    99+
    2024-04-02
  • Js中async/await的示例分析
    这篇文章将为大家详细讲解有关Js中async/await的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。async/await 是一种编写异步代码的新方法。之前异...
    99+
    2024-04-02
  • MySQL不再支持Berkeley DB 转而添加插件的示例分析
    这篇文章将为大家详细讲解有关MySQL不再支持Berkeley DB 转而添加插件的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。 在最新的5.1.12版中,My...
    99+
    2024-04-02
  • webpack2+React的示例分析
    这篇文章主要为大家展示了“webpack2+React的示例分析”,内容简而易懂,条理清晰,希望能够帮助大家解决疑惑,下面让小编带领大家一起研究并学习一下“webpack2+React的示例分析”这篇文章吧...
    99+
    2024-04-02
  • react hooks的示例分析
    这篇文章将为大家详细讲解有关react hooks的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。React在16.8版本正式发布了Hooks。关注了很久,最近正...
    99+
    2024-04-02
  • JavaScript中$(function(){})写法的示例分析
    小编给大家分享一下JavaScript中$(function(){})写法的示例分析,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!$(function(){})写...
    99+
    2023-06-03
  • C#支持事件举例分析
    本篇内容介绍了“C#支持事件举例分析”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!这里介绍C#支持事件(这个特点也是MSVJ所具有的),当前...
    99+
    2023-06-17
  • 支持python分布式计算框架Ray的示例分析
    这篇文章将为大家详细讲解有关支持python分布式计算框架Ray的示例分析,小编觉得挺实用的,因此分享给大家做个参考,希望大家阅读完这篇文章后可以有所收获。1、简介Ray为构建分布式应用程序提供了一个简单、通用的API。Ray是一种分布式执...
    99+
    2023-06-20
  • Javascript之React的示例分析
    这篇文章主要介绍Javascript之React的示例分析,文中介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们一定要看完!前言React核心的单向数据流、一切皆数据的state、不会改变的props,...
    99+
    2024-04-02
  • React中ref的示例分析
    这篇文章给大家分享的是有关React中ref的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。为了摆脱繁琐的Dom操作, React提倡组件化, 组件内部用数据来驱动视图的...
    99+
    2024-04-02
  • HTML5浏览器支持实例分析
    本篇内容主要讲解“HTML5浏览器支持实例分析”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“HTML5浏览器支持实例分析”吧! HTML5 浏览器支持 所有...
    99+
    2024-04-02
  • Perf支持Zstd压缩跟踪记录的示例分析
    这篇文章给大家介绍Perf支持Zstd压缩跟踪记录的示例分析,内容非常详细,感兴趣的小伙伴们可以参考借鉴,希望对大家能有所帮助。Linux内核的perf子系统的后期更新引入了对跟踪压缩记录的支持,这可以使文件大小减少三到五倍。Linux内核...
    99+
    2023-06-05
  • Ubuntu Touch支持USB Tethering上网功能的示例分析
    Ubuntu Touch支持USB Tethering上网功能的示例分析,针对这个问题,这篇文章详细介绍了相对应的分析和解答,希望可以帮助更多想解决这个问题的小伙伴找到更简单易行的方法。关于 USB Tethering 功能介绍,该功能是电...
    99+
    2023-06-16
  • jvm支持最大线程数测试的示例分析
    这篇文章给大家分享的是有关jvm支持最大线程数测试的示例分析的内容。小编觉得挺实用的,因此分享给大家做个参考,一起跟随小编过来看看吧。最近想测试下Openfire下的最大并发数,需要开大量线程来模拟客户端。对于一个JVM实例到底能开多少个线...
    99+
    2023-05-30
    jvm
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作