Vue3 实现一个自带防抖的响应式值 —— CustomRef

前言

写一个防抖函数自然难不倒各位同学。但实现一个自带防抖 的响应式值大家可否听说过,本文使用vue3中的customRef来实现。

基本介绍

customRef 函数可以创建一个自定义ref,并对其依赖项跟踪和更新触发进行控制。该函数接受一个工厂函数作为参数,该工厂函数接收track和trigger两个函数作为参数。

了解过Vue3原理的同学都知道track主要负责get时进行依赖收集,trigger主要负责通知更新

废话不多说,我们直接看如何使用customRef,优雅的实现一个自带防抖的响应式值

例子

js 复制代码
<template>
  {{ value }}
  <input type="text" placeholder="请输入内容..." v-model="value" />
</template>
<script lang="ts" setup>
import { customRef, ref } from "vue";
let time: any = null;
const value = customRef((track, trigger) => {
  let defaultValue = "默认值";
  return {
    get() {
      track();
      return defaultValue;
    },
    set(newValue) {
      clearTimeout(time);
      time = setTimeout(() => {
        defaultValue = newValue;
        console.log("value update");
        trigger();
      }, 500);
    },
  };
});
</script>

上述代码中,我们使用customRef,定义了一个响应式的值,并且使用v-model进行双向绑定。其内部声明了一个闭包变量defaultValueget时调用track进行依赖收集的同时返回,set时修改defaultValue,并调用trigger进行更新,其内部也是利用setTimeOut实现防抖。

输入框进行输入的时候,value只会更新一次。

以上就是customRef实现一个自带防抖的响应式值。当然我们也可以尊重一下CompositionAPI:

js 复制代码
// useDebounceRef.ts
import {customRef} from "vue";
export function useDebounceRef(defaultValue:any,delay = 300){
    let time: any = null;
    return customRef((track, trigger) => {
        return {
          get() {
            track();
            return defaultValue;
          },
          set(newValue) {
            clearTimeout(time);
            time = setTimeout(() => {
              defaultValue = newValue;
              trigger();
            }, 500);
          },
        };
      })
}

// 组件中使用
<template>
  <div>
    {{ value }}
    <input type="text" placeholder="请输入内容..." v-model="value" />
  </div>
</template>
<script lang="ts" setup>
import { useDebounceRef } from "./useDebounceRef";
const value = useDebounceRef("默认值");
</script>
相关推荐
bin915313 分钟前
npm报错
前端·npm·node.js
一指流沙q20 分钟前
Chrome被360导航篡改了怎么改回来?
前端·chrome
laocooon5238578861 小时前
HTML CSS 超链
前端·css·html
LUwantAC1 小时前
CSS(二):美化网页元素
前端·css
m0_748251081 小时前
docker安装nginx,docker部署vue前端,以及docker部署java的jar部署
java·前端·docker
我是ed1 小时前
# thingjs 基础案例整理
前端
Ashore_1 小时前
从简单封装到数据响应:Vue如何引领开发新模式❓❗️
前端·vue.js
落魄实习生1 小时前
小米路由器开启SSH,配置阿里云ddns,开启外网访问SSH和WEB管理界面
前端·阿里云·ssh
顽疲1 小时前
从零用java实现 小红书 springboot vue uniapp (6)用户登录鉴权及发布笔记
java·vue.js·spring boot·uni-app
bug丸2 小时前
v8引擎垃圾回收
前端·javascript·垃圾回收