自定义一个vue useFetch

自定义hooks

在 vue2 里面有一些公共的逻辑,我们可以使用 mixins(混入)。vue3 里面的逻辑复用,官方推荐的方式包括自定义组合式函数,也就是自定义一些hooks,官方案例

vue官方也有一些好用的 hooks 工具集,vueuse 其中的一些 hook 在 nuxt3 里面也是开箱即用,例如 useFetch。

useFetch 可以发送一个 fetch 请求,并且返回响应式的数据。简化我们创建响应式数据并且请求赋值的流程,非常的方便。

自定义一个 useFetch

通常我们的前端项目都使用 axios 来封装发送请求,但是 useFetch 里面发送请求的却是其他的库 ofetch,具体可以查看 nuxt3 里面 useFetch 的说明。

所以我就封装了一个基于自己的项目的 useFetch 处理请求的方法。代码如下:

js 复制代码
import {ref, watch} from 'vue';
import {ElMessage} from "element-plus";

export const useFetch = (fetch, {
    immediate = true, // 是否立即发送请求
    effectArray = [], // 副作用依赖项,watch 监听,以便重新发送请求
    beforeFetch, // 请求发送之前执行的回调
    afterFetch, // 请求发送之后执行的回调
    onSuccess, // 成功的回调函数
    onError, // 错误的回调函数
    updateDataOnError = false, // 接口出错时是否更新数据
    initialData = null // 初始化数据响应式数据,可用于页面渲染
} = {
    immediate: true,
    updateDataOnError: false,
    effectArray: [],
    initialData: null
}) => {

    const isLoading = ref(false) // 是否正在加载中
    const isFinished = ref(false) // 请求是否结束
    const data = ref(initialData) // 根据 initialData 创建的响应式的数据
    const response = ref(null) // 响应式的接口返回的 response
    const error = ref(null) // 响应式的请求错误数据
    
    // 执行请求的方法,方便手动控制
    const execute = () => {
        beforeFetch && beforeFetch()
        isLoading.value = true
        isFinished.value = false
        error.value = null
        fetch().then(res => {
            response.value = res
            if (res?.status === '00000') {
                onSuccess && onSuccess(res)
                data.value = res.data
            } else {
                ElMessage.error(res?.message)
                if (updateDataOnError) {
                    data.value = null
                }
            }
        }).catch(e => {
            onError && onError(e)
            error.value = e
        }).finally(() => {
            afterFetch && afterFetch()
            isLoading.value = false
            isFinished.value = true
        })
    }

    if (immediate) {
        execute()
    }

    // 监听依赖项发生变化,重新发起请求
    if (effectArray.length > 0) {
        watch(effectArray, execute)
    }

    return {
        isLoading,
        isFinished,
        response,
        error,
        data,
        execute
    }
}

使用案例:

js 复制代码
const page = reactive({
    pageNum: 1,
    pageSize: 10
})

// 假设需要响应式的 table 列表数据
const initialData = reactive({
    list: [],
    total: 0
})

// 此处省略了 axios 实例的一些配置
const {data} = useFetch(() => axios.post('/getList',{pageNum: page.pageNum,pageSize: page.pageSize}), {
    initialData,
    immediate: false,
    effectArray: [() => page.pageNum,() => page.pageSize]
})

大家可以根据自己的项目需求修改一些配置。 类似的也可以根据 react 里面 useState, useEffect 创建一个 react 版本的 useFetch。ahooks 也是不错的导师,里面可以学到很多思路。

相关推荐
天天摸鱼的java工程师1 分钟前
互联网行业能力解刨:从Java后端八年开发经验看
前端·后端·程序员
FlyingBird~6 分钟前
CocosCreator 之 JavaScript/TypeScript和Java的相互交互
java·javascript·typescript
brzhang8 分钟前
Android 16 卫星连接 API 来了,带你写出「永不失联」的应用
前端·后端·架构
John_ToDebug22 分钟前
Chrome 浏览器前端与客户端双向通信实战
前端·c++·chrome
要加油哦~27 分钟前
CSS | transition 和 transform的用处和区别
前端·css
小鱼人爱编程38 分钟前
现代大前端是如何编码的?
android·前端·flutter
神仙别闹41 分钟前
基于Java+VUE+MariaDB实现(Web)仿小米商城
java·前端·vue.js
袁煦丞1 小时前
低成本私有云存储方案Nas-Cab:cpolar实验室第508次成功挑战
前端·程序员·远程工作
小公主1 小时前
「前端必备」Flex 布局全解析:从入门到深度计算,搞懂弹性盒子!
前端·css
江城开朗的豌豆1 小时前
前端性能救星!用 requestAnimationFrame 丝滑渲染海量数据
前端·javascript·面试