首页 > 其他分享 >牛逼!在Vue3.5中仅仅2分钟就能封装一个自动cancel的fetch函数

牛逼!在Vue3.5中仅仅2分钟就能封装一个自动cancel的fetch函数

时间:2024-09-12 16:51:31浏览次数:3  
标签:函数 代码 watch controller Vue3.5 onWatcherCleanup cancel 回调 fetch

前言

在欧阳的上一篇 这应该是全网最详细的Vue3.5版本解读文章中有不少同学对Vue3.5新增的onWatcherCleanup有点疑惑,这个新增的API好像和watch API回调的第三个参数onCleanup功能好像重复了。今天这篇文章来讲讲新增的onWatcherCleanup函数的使用场景:封装一个自动cancel的fetch函数

加入欧阳的高质量vue源码交流群、欧阳平时写文章参考的多本vue源码电子书

watch回调的第三个参数onCleanup

有些同学可能还不清楚watch回调的第三个参数onCleanup,我们先来看个demo,代码如下:


javascript

代码解读

复制代码

watch(id, (value, oldValue, onCleanup) => { console.log("do something"); onCleanup(() => { console.log("cleanup"); }); });

watch回调的前两个参数大家应该很熟悉,分别是value新的值,oldValue旧的值。

第三个参数onCleanup大家平时可能用的不多,这是一个回调函数,当watch的值改变后或者组件销毁前就会执行onCleanup传入的回调。

在上面的demo中就是变量id改变时会触发onCleanup中的回调,进而console打印"cleanup"字符串。又或者所在的组件销毁前也会触发onCleanup中的回调,进而console打印"cleanup"字符串。

那我们在onCleanup中可以干嘛呢?

答案是可以清理副作用,比如在watch中使用setInterval初始化一个定时器。那么我们就可以在onCleanup的回调中清理掉定时器,无需去组件的beforeUnmount钩子函数去统一清理。

onWatcherCleanup函数

onWatcherCleanup函数的作用和watch回调的第三个参数onCleanup差不多,也是当watch的值改变后或者组件销毁前就会执行onWatcherCleanup传入的回调。

使用方法也很简单,代码如下:


javascript

代码解读

复制代码

import { watch, onWatcherCleanup } from "vue"; watch(id, () => { console.log("do something"); onWatcherCleanup(() => { console.log("cleanup"); }); });

从上面的代码可以看到onWatcherCleanup的用法其实和watch回调的第三个参数onCleanup差不多,区别在于这里的onWatcherCleanup是从vue中import导入的。

除了从vue中import导入的区别以外,还有一个区别是onWatcherCleanup不光在watch中可以使用,在watchEffect中同样也可以使用。比如下面这样的:


javascript

代码解读

复制代码

watchEffect(() => { console.log("do something in watchEffect", id.value); onWatcherCleanup(() => { console.log("cleanup watchEffect"); }); });

和前面的例子一样,上面的代码中id的值改变后或者组件销毁时也会执行onWatcherCleanup函数中的console.log打印。

onWatcherCleanup函数是从vue中import导入的,那么这意味着onWatcherCleanup函数的调用可以写在任意地方,只要最终经过函数的层层调用后还是在watch或者watchEffect的回调中就可以。

利用上面的这一特点我们可以使用onWatcherCleanup做到一些onCleanup做不到的事情,比如:封装一个自动cancelfetch函数。

封装自动cancel的fetch函数

在讲这个之前我们先来了解一下如何cancel一个fetch函数。

这里涉及到AbortController接口,AbortController 接口表示一个控制器对象,允许你根据需要中止一个或多个 Web 请求。

下面这个是cancel取消一个请求的demo,代码如下:


javascript

代码解读

复制代码

const controller = new AbortController(); const res = await fetch(url, { ...options, signal: controller.signal, }); setTimeout(() => { controller.abort(); }, 500);

首先使用new AbortController()创建一个控制器对象controller

其中的controller.signal返回一个 AbortSignal 对象实例,可以用它来和异步操作进行通信或者中止这个操作。

在我们这里把controller.signal作为signal选项直接传给fetch函数就可以了。

最后就是可以使用controller.abort()将fetch请求取消掉,在上面的demo中是如果超过500ms请求还没完成,那么就执行controller.abort()将fetch请求取消掉。

有了前面的知识铺垫,我们先来看看使用“自动cancelfetch函数”的地方,代码如下:


javascript

代码解读

复制代码

<script setup lang="ts"> import { watch, ref, watchEffect, onWatcherCleanup } from "vue"; import myFetch from "./myFetch"; const id = ref(1); const data = ref(null); watch(id, async () => { const res = await myFetch(`http://localhost:3000/api/${id.value}`, { method: "GET", }); console.log(res); data.value = res; }); </script> <template> <p>data is: {{ data }}</p> <button @click="id++">id++</button> </template>

在上面的例子中使用watch监听了变量id,在监听的回调中会使用封装的myFetch函数请求接口。

上面的例子大家平时应该经常遇到,如果id的值变化很快,但是服务端接口请求需要2秒才能完成,这时我们期望只有最后一次id的值改变触发的请求才需要完成,其他请求都cancel取消掉。

如果在myFetch请求的过程中组件被销毁了,此时我们也期望能够将请求cancel取消掉。

在Vue3.5之前想要去实现上面的这两个需求很麻烦,但是有了Vue3.5的onWatcherCleanup函数后就非常容易了。

这个是封装的自动cancelfetch函数,myFetch.ts文件代码如下:


javascript

代码解读

复制代码

import { getCurrentWatcher, onWatcherCleanup } from "vue"; export default async function myFetch(url: string, options: RequestInit) { const controller = new AbortController(); if (getCurrentWatcher()) { onWatcherCleanup(() => { controller.abort(); }); } const res = await fetch(url, { ...options, signal: controller.signal, }); let json; try { json = await res.json(); } catch (error) { json = { code: 500, message: "JSON format error", }; } return json; }

由于onWatcherCleanup函数是从vue中import导入,那么我们就可以在自己封装的myFetch函数中导入和使用他。

onWatcherCleanup函数的回调中我们执行了controller.abort(),前面已经讲过了当watch或者watchEffect的回调执行前或者组件卸载前就会执行里面的onWatcherCleanup注册的回调。我们这里的myFetch是在watch中调用的,当然也会触发里面的onWatcherCleanup注册的回调。

onWatcherCleanup的回调中执行了controller.abort(),前面我们讲过了执行controller.abort()就会将正在请求的fetch函数给cancel取消掉。

就这么简单的就实现了前面的两个需求:

需求一:**如果id的值变化很快,但是服务端接口请求需要2秒才能完成,这时我们期望只有最后一次id的值改变触发的请求才需要完成,其他请求都cancel取消掉。**下面这个是变量id在短时间内多次修改的gif效果图:

从上面的gif图可以看到只有最后一个请求是完成了的,其他请求全部被cancel掉。

需求二:**如果在myFetch请求的过程中组件被销毁了,此时我们也期望能够将请求cancel取消掉。**下面这个是组件卸载时gif效果图:

从上图中可以看到在卸载组件时组件正在从服务端请求数据,此时请求会自动cancel掉。

细心的小伙伴发现了在myFetch函数中,onWatcherCleanup函数外面套了一个getCurrentWatcher的判断,代码如下:


javascript

代码解读

复制代码

import { getCurrentWatcher, onWatcherCleanup } from "vue"; export default async function myFetch(url: string, options: RequestInit) { // ...省略 if (getCurrentWatcher()) { onWatcherCleanup(() => { controller.abort(); }); } // ...省略 }

当watch或者watchEffect监听的值改变后onWatcherCleanup的回调就会触发,所以onWatcherCleanup的执行是由其所在的watch或者watchEffect触发的。

如果onWatcherCleanup不在watch或者watchEffect的回调中执行,那么当然onWatcherCleanup中的回调也永远不会执行。

可能有的小伙伴有疑问,你这里的onWatcherCleanup是在myFetch中执行的,也没在watch或者watchEffect的回调中执行吖?

答案是myFetch函数的执行是在watch中执行的,myFetch然后再去执行onWatcherCleanup

getCurrentWatcher()函数就会返回当前正在执行回调的watch或者watchEffect,如果当前myFetch不是在watch或者watchEffect的回调中执行的,那么getCurrentWatcher()函数的返回值就是空,所以这种情况就不需要去执行onWatcherCleanup函数了。

最后值得一提的是onWatcherCleanup不能在await后面执行,比如下面这样的代码:


javascript

代码解读

复制代码

import { getCurrentWatcher, onWatcherCleanup } from "vue"; export default async function myFetch(url: string, options: RequestInit) { const controller = new AbortController(); const res = await fetch(url, { ...options, signal: controller.signal, }); let json; try { json = await res.json(); } catch (error) { json = { code: 500, message: "JSON format error", }; } // ❌ 错误的写法 if (getCurrentWatcher()) { onWatcherCleanup(() => { controller.abort(); }); } return json; }

在上面的代码中我们将onWatcherCleanup调用放在了await fetch()的后面,这种写法onWatcherCleanup注册的回调是不会执行的

为什么在await后面的onWatcherCleanup注册的回调永远不会执行呢?

答案是js的await相当于注册了一个回调函数去执行await后的代码,当await等待结束后再去执行这个回调函数,从而执行await后的代码。

await以及之前的代码确实是在watch回调中执行的,我们这里的onWatcherCleanup就是await后面的代码,await后面的代码是在一个新的回调中执行的,也就是watch“回调中”的“回调中”执行的。

onWatcherCleanup执行时已经不知道当前正在执行的watch回调是谁了,所以onWatcherCleanup的回调也没注册上。当watch的变量修改时或者组件卸载时onWatcherCleanup注册的回调永远也不会执行。

总结

watch或者watchEffect监听的变量修改时,以及组件卸载时,会去执行他们回调中使用onWatcherCleanup注册的回调函数。并且onWatcherCleanup是从vue中import导入的,使得我的可以在任意地方执行onWatcherCleanup函数。利用这两个特性我们就可以封装一个自动cancel的fetch函数。

最后推荐一下欧阳自己写的开源电子书vue3编译原理揭秘,看完这本书可以让你对vue编译的认知有质的提升,并且这本书初、中级前端能看懂。完全免费,只求一个star。

原文链接:https://juejin.cn/post/7412852276958183461

标签:函数,代码,watch,controller,Vue3.5,onWatcherCleanup,cancel,回调,fetch
From: https://blog.csdn.net/weixin_47588164/article/details/142134735

相关文章

  • Express和Node中使用Fetch POST数据到服务器时遇到405错误
    和Node中使用FetchPOST数据到服务器时遇到405错误一、问题描述(一)错误现象在Express和Node中使用Fetch发送POST请求时,服务器返回了405错误。具体表现为,客户端向服务器发送POST请求后,服务器没有正确处理请求,而是返回了405错误代码,表示方法不被允许。这种错误通......
  • 使用Vue3.5的onWatcherCleanup封装自动cancel的fetch函数
    前言在欧阳的上一篇这应该是全网最详细的Vue3.5版本解读文章中有不少同学对Vue3.5新增的onWatcherCleanup有点疑惑,这个新增的API好像和watchAPI回调的第三个参数onCleanup功能好像重复了。今天这篇文章来讲讲新增的onWatcherCleanup函数的使用场景:封装一个自动cancel的fetch函......
  • 前端使用Fetch 后端Spring Boot实现下载文件(浏览器下载栏)
    前后端分别实现前端代码后端代码前端代码constdownloadFile=()=>{fetch("/getDocx?fileName=文件名称.docx").then((response)=>{console.log('response:>>',response);if(!response.ok){thrownewError("本次请求......
  • 【Java】已解决:java.util.concurrent.CancellationException
    文章目录一、分析问题背景二、可能出错的原因三、错误代码示例四、正确代码示例五、注意事项已解决:java.util.concurrent.CancellationException一、分析问题背景java.util.concurrent.CancellationException是一种常见的运行时异常,当尝试访问已取消的任务时......
  • 这应该是全网最详细的Vue3.5版本解读
    版本号这次的版本号是天元突破红莲螺岩,这是07年出的一个二次元动漫,作者是没看过的。在此之前我一直以为这次的版本号会叫黑神话:悟空,可能悟空不够二次元吧。响应式响应式相关的内容主要分为:重构响应式、响应式props支持解构、新增onEffectCleanup函数、新增basewatch函数......
  • 这应该是全网最详细的Vue3.5版本解读kh
    合集-vue3代码修炼秘籍(16)1.答应我,在vue中不要滥用watch好吗?02-292.一文搞懂Vue3defineModel双向绑定:告别繁琐代码!02-043.没有虚拟DOM版本的vue(VueVapor)01-264.有了CompositionAPI后,有些场景或许你不需要pinia了01-235.你不知道的vue3:使用runWithContext实现在非setup期......
  • 这应该是全网最详细的Vue3.5版本解读
    前言Vue3.5正式版在这两天发布了,网上已经有了不少关于Vue3.5版本的解读文章。但是欧阳发现这些文章对3.5中新增的功能介绍都不是很全,所以导致不少同学有个错觉,觉得Vue3.5版本不过如此,选择跳过这个版本等下个大版本再去更新。所以欧阳写了这篇超级详细的Vue3.5版本解读文章,小伙伴......
  • 牛逼!Vue3.5的useTemplateRef让ref操作DOM更加丝滑
    前言vue3中想要访问DOM和子组件可以使用ref进行模版引用,但是这个ref有一些让人迷惑的地方。比如定义的ref变量到底是一个响应式数据还是DOM元素?还有template中ref属性的值明明是一个字符串,比如ref="inputEl",怎么就和script中同名的inputEl变量绑到一块了呢?所以Vue3.5推出了一个us......
  • javascript基础从小白到高手系列一千九百二十四:Fetch API
    FetchAPI能够执行XMLHttpRequest对象的所有任务,但更容易使用,接口也更现代化,能够在Web工作线程等现代Web工具中使用。XMLHttpRequest可以选择异步,而FetchAPI则必须是异步。FetchAPI是WHATWG的一个“活标准”(livingstandard),用规范原文说,就是“Fetch标准定义请......
  • preload、prefetch
    preloadpreload是一种声明式的资源预加载技术,它告诉浏览器页面即将需要的资源,并请求浏览器提前加载这些资源。preload的主要特点包括:高优先级:preload加载的资源通常具有较高的优先级,浏览器会尽可能快地加载这些资源,但加载过程不会阻塞文档的解析或window的onload事件。指定资......