vue3.x系列之封装响应式的hooks技巧

v3.x封装一个接口请求的hooks的,都应该很熟练了,那么要实现一个响应式的hooks该怎么做?

实现效果

代码实现

js 复制代码
<template>
  <div v-if="error">Oops! Error encountered: {{ error.message }}</div>
  <div v-else-if="data">
    Data loaded:
    <pre>{{ data }}</pre>
  </div>
  <div v-else>Loading...</div>
  <button @click="url = 'https://jsonplaceholder.typicode.com/todos/2'">发送</button>
</template>

<script setup>
import { useFetch } from "@/hooks/useFetch";
import { ref } from "vue";
let url = ref("https://jsonplaceholder.typicode.com/todos/1");
const { data, error } = useFetch(url);
</script>
js 复制代码
import { delay } from "lodash-es";
import { ref, watchEffect, toValue } from "vue";

export function useFetch(url) {
  const data = ref(null);
  const error = ref(null);
  function fetchData() {
    fetch(toValue(url))
      .then((res) => res.json())
      .then((res) => {
        delay(() => {
          data.value = res;
        }, 2000);
      })
      .catch((err) => (error.value = err));
  }
  watchEffect(() => {
    fetchData();
  });
  return { data, error };
}

技巧总结

  • 这里我们传入的是一个url接口的ref,不是普通的字符串数据
  • hooks里面接收的时候,接口请求使用toValue转为普通的字符数据
  • watchEffect监听副作用,当ref数据变化时,及时响应来发起新的请求
  • delay方法模拟接口请求延时效果,区分出请求中的状态

v3.x的几个工具方法isRef,unref,toRef的对比

js 复制代码
// isRef  判断是否是ref类型
let url = ref("https://jsonplaceholder.typicode.com/todos/1");
const str = "hello world";
console.log(isRef(url), "isRef");
console.log(isRef(str), "isRef");
js 复制代码
// unref 解包ref
const unRefUrl = unref(url);
console.log("unRefUrl", unRefUrl);
console.log(unref(str), "unRefStr");
js 复制代码
// toRef 包装ref,可以将值、refs 或 getters 规范化为 refs (3.3+)。
const toRefUrl = toRef(url);
console.log("toRefUrl", toRefUrl);
console.log(toRef(str), "toRefStr");
js 复制代码
const state = reactive({
  foo: 1,
  bar: 2
});
const foo = toRef(state, "foo");

const foo1 = ref(state.foo);

上面这个 ref 不会和 state.foo 保持同步,因为这个 ref() 接收到的是一个纯数值。

  • toRef和ref的区别如下图
相关推荐
前端大聪明200218 小时前
single-spa原理解析
前端·javascript
一枚前端小能手18 小时前
📦 从npm到yarn到pnpm的演进之路 - 包管理器实现原理深度解析
前端·javascript·npm
影i18 小时前
CSS Transform 和父元素撑开问题
前端
@大迁世界18 小时前
Promise.all 与 Promise.allSettled:一次取数的小差别,救了我的接口
开发语言·前端·javascript·ecmascript
知识分享小能手18 小时前
微信小程序入门学习教程,从入门到精通,项目实战:美妆商城小程序 —— 知识点详解与案例代码 (18)
前端·学习·react.js·微信小程序·小程序·vue·前端技术
DoraBigHead19 小时前
React 中的代数效应:从概念到 Fiber 架构的落地
前端·javascript·react.js
LuckySusu19 小时前
【vue篇】Vue 性能优化全景图:从编码到部署的优化策略
前端·vue.js
卓伊凡19 小时前
【03】建立隐私关于等相关页面和内容-vue+vite开发实战-做一个非常漂亮的APP下载落地页-支持PC和H5自适应提供安卓苹果鸿蒙下载和网页端访问-优雅草卓
前端
笨笨鸟慢慢飞19 小时前
Vue3后退不刷新,前进刷新
前端
LuckySusu19 小时前
【vue篇】SSR 深度解析:服务端渲染的“利”与“弊”
前端·vue.js