watchEffect

执行传入的一个函数,同时自动追踪函数中依赖到的数据,并在其依赖变更时重新运行该函数。
并且会在 组件挂载前 立即调用一次,(默认是挂载前,可通过修改 flush 属性改变,后边会讲到)
例子
const person = reactive({
count: 0,
});
onBeforeMount(() => {
console.log("onBeforeMount");
});
onMounted(() => {
console.log("onMounted");
});
watchEffect(
() => { const el = document.querySelector(".h1");
console.log(el);
console.log(person.count);
}
);

停止监听器
调用返回值会停止监听,组件卸载时也会自动停止监听 (异步创建的监听器,组件卸载是不会自动停止监听的,需要手动调用,防止内存泄漏)
例子:
const stop = watchEffect(
() => {
console.log(person.count);
}
);
stop ()

// 注意!异步创建,组件卸载不会停止监听
setTimeout(() => {
watchEffect(() => {})
}, 100)
 
清除副作用
有时副作用函数会执行一些异步的副作用,需要在失效时清除(完成前状态已改变)。
可用调用 onInvalidate 函数。
副作用函数接收一个 onInvalidate 函数作为参数。 onInvalidate 接收一个回调函数。
第一次自调用 effect 副作用函数 是不会触发 onInvalidate 函数的,
触发 onInvalidate 函数有以下情况
依赖数据改变,再次调用 effect 副作用函数,
监听器被停止 (组件卸载前也会调用,因为组件卸载就会停止监听器)
例子
watchEffect(
(onInvalidate) => {
onInvalidate(() => {
console.log("before"
);
});

const el = document.querySelector(".h1");
console.log(el);
console.log(person.count);
}
);

第一次运行是不会触发

改变数据依赖时触发

组件卸载前触发

 副作用刷新时机

vue 默认会缓存副作用函数,异步刷新,多个依赖数据同时改变也只会触发一次。并且会监听组件更新函数,在组件更新前去触发副作用函数。

可给 watchEffect 传递第二个参数,options 配置对象,改变 副作用函数触发时机,和执行次数。

options 配置对象中,flush 属性有三个值,'pre'、'post' 或 'sync'

pre :默认配置,组件挂载前 和 组件更新前 执行副作用函数,并且会缓存副作用函数,异步刷新,同时改变多个依赖数据只会调用一次副作用函数。

post : 组件挂载后 和 组件更新后 执行副作用函数,并且会缓存副作用函数,异步刷新,同时改变多个依赖数据只会调用一次副作用函数。

sync: 组件挂载前 和 组件更新前 执行副作用函数,不会缓存副作用函数,同步刷新,同时改变多个依赖数据会多次调用副作用函数。少用sync

默认配置,例子:

const state = ref(0);
const person = reactive({
count: 0,
});
const change2 = () => {
state.value++;
person.count++;
};
watchEffect(
(onvalidate) => {
onvalidate(() => {
console.log("before");
});
const el = document.querySelector(".h1");
console.log(el);
console.log(person.count);
console.log(state.value);
}
);

默认情况下 副作用函数会在 组件挂载前 和 组件更新前 执行,并且会缓存副作用函数,异步刷新,同时改变多个依赖数据只会调用一次副作用函数

flush 属性值为 post,例子:

watchEffect(
(onvalidate) => {
onvalidate(() => {
console.log("before");
});
const el = document.querySelector(".h1");
console.log(el);
console.log(person.count);
console.log(state.value);
},
{
flush:
"post"
}

);

在 组件挂载后,和 组件更新后 执行,并且会缓存副作用函数,异步刷新,同时改变多个依赖数据只会调用一次副作用函数

flush 属性值为 sync,例子:

watchEffect(
(onvalidate) => {
onvalidate(() => {
console.log("before");
});
const el = document.querySelector(".h1");
console.log(el);
console.log(person.count);
console.log(state.value);
},
{
flush:
"sync"
}

);

在 组件挂载前,和 组件更新前 执行,不会缓存副作用函数,同步刷新,同时改变多个依赖数据就多次调用副作用函数

监听器调试

监听器配置对象中,onTrack 和 onTrigger  选项可用于调试监听器行为(开发模式)

onTrack:会在响应式数据被追踪的时候被调用。(就是数据被获取时)

onTrigger: 响应式依赖改变时被调用

例子

const state = ref(0);
const person = reactive({
count: 0,
});
const change2 = () => {
state.value++;
person.count++;
};
watchEffect(
(onvalidate) => {
onvalidate(() => {
console.log("before");
});
const el = document.querySelector(".h1");
console.log(el);
console.log(state.value);
console.log(person.count);
},
{
onTrack(e) {
console.log(e,
"track");
},
onTrigger(e) {
console.log(e, "trigger"
);
},
}

);

第一次自动执行,获取响应式数据依赖,触发 onTrack

改变依赖数据,触发 onTrigger

watch

监听特定的数据源,在回调函数中执行副作用,默认只有数据改变才会执行副作用函数, 不会立即调用副作用函数(可配置)

和 watchEffect 相比

  • 能访问监听数据变化前后值
  • 手动添加声明监听哪些数据
  • 也有 onInvalidate, 会作为回调函数的第三个参数,(注意,副作用函数第二次执行才会调用 onInvalidate)
  • 配置选项 flush 等.. watch 都有,多了 deep选项和 immediate 选项

监听单个数据源,例子

const state = ref(0);
const person = reactive({
count: 0,
});
const change2 = () => {
state.value++;
person.count++;
};
// 侦听一个 getter
watch(
() => person.count,
(count, prevCount) => {
console.log(count, "count");
console.log(prevCount, "prevCount");
}
); // 直接侦听ref
watch(state, (state, prevState) => {
console.log(state, "count");
console.log(prevState, "prevCount");
});

监听多个数据源,并且配置选项,例子

const state = ref(0);
const person = reactive({
count: 0,
});
const change2 = () => {
state.value++;
person.count++;
};
watch(
[() => person.count, state],
(arr, prevArr, onvalidate) => {
console.log(arr, "arr");
console.log(prevArr, "prevArr");
onvalidate(() => {
console.log("before");
});
},
{
immediate: true,
flush: "pre",
onTrigger(e) {
console.log(e, "trigger");
},
onTrack(e) {
console.log(e, "track");
},
}
);

监听多个数据源用数组写法,回调函数参数1和2 也变成数组,配置选项 immediate 为 ture 时,初始化时会自执行一次副作用函数, 其它的配置项和 watchEffect 没区别

监听对象或数组,什么情况下配置 deep 选项

数据类型 直接监听 getter 写法 克隆数据,必须采用 getter 写法
ref 需要deep 需要deep 不需要deep
reactive 不需要deep 需要deep 不需要deep

(注意,响应式数据为对象时直接监听,用的是同一个对象的引用,导致回调函数参数1和2,改变前后数据相同)。所以要克隆一个副本监听此副本,并使用 getter 写法。

例子:

const person = ref({
name: { Anna: { count: 0 } },
});
const person2 = reactive({
name: { Anna: { count: 0 } },
});
const change2 = () => {
person.value.name.Anna.count++;
person2.name.Anna.count++;
};
// 直接监听 ref
watch(
person,
(person, prevPerson) => {
console.log(person.name.Anna, "person");
console.log(prevPerson.name.Anna, "prevPerson");
},
{
deep: true,
}
);
// 直接监听 reactive
watch(person2, (person, prevPerson) => {
console.log(person.name.Anna, "person");
console.log(prevPerson.name.Anna, "prevPerson");
});
// 监听reactive,getter 写法
watch(
() => person2,
(person, prevPerson) => {
console.log(person.name.Anna, "person");
console.log(prevPerson.name.Anna, "prevPerson");
},
{
deep: true,
}
);

回调函数参数 1和2 出现数据相同 的问题

克隆响应式数据再监听,需要 getter 写法

const person = reactive({
name: { Anna: { count: 0 } },
});
const change2 = () => {
person.name.Anna.count++;
};
watch(
() => deepClone(person),
(person, prevPerson) => {
console.log(person.name.Anna, "person");
console.log(prevPerson.name.Anna, "prevPerson");
}
);

现在 回调函数参数1和2,改变前后数据就正确了

建议:响应式数据为对象或数组用watch监听时,用 getter 写法。

vue3 watch笔记的更多相关文章

  1. vue3 学习笔记 (二)——axios 的使用有变化吗?

    本篇文章主要目的就是想告诉我身边,正在学 vue3 或者 准备学 vue3 的同学,vue3中网络请求axios该如何使用,防止接触了一点点 vue3 的同学会有个疑问?生命周期.router .vu ...

  2. vue3 学习笔记 (四)——vue3 setup() 高级用法

    本篇文章干货较多,建议收藏! 从 vue2 升级到 vue3,vue3 是可以兼容 vue2 的,所以 vue3 可以采用 vue2 的选项式API.由于选项式API一个变量存在于多处,如果出现问题时 ...

  3. vue3 学习笔记(九)——script setup 语法糖用了才知道有多爽

    刚开始使用 script setup 语法糖的时候,编辑器会提示这是一个实验属性,要使用的话,需要固定 vue 版本. 在 6 月底,该提案被正式定稿,在 v3.1.3 的版本上,继续使用但仍会有实验 ...

  4. vue3 学习笔记 (五)——vue3 的 setup 如何实现响应式功能?

    setup 是用来写组合式 api ,内部的数据和方法需要通过 return 之后,模板才能使用.在之前 vue2 中,data 返回的数据,可以直接进行双向绑定使用,如果我们把 setup 中数据类 ...

  5. vue3 高阶 API 大汇总,强到离谱

    高阶函数是什么呢? 高阶函数英文名叫:Higher Order function ,一个函数可以接收一个或多个函数作为输入,或者输出一个函数,至少满足上述条件之一的函数,叫做高阶函数. 前言 本篇内容 ...

  6. Vue3 全家桶,从 0 到 1 实战项目,新手有福了

    前端发展百花放,一技未熟百技出.未知何处去下手,关注小编胜百书. 我是前端人,专注分享前端内容! 本篇文章主要是,使用 vite 创建一个vue3 项目,实践 vie-router4 vuex4 结合 ...

  7. 尤雨溪在直播中讲到的Vue3.0 Beta的那些特性,快记笔记了

    前言 在那天风雨交加的夜晚,Vue的创作者尤雨溪尤大大在b站直播分享了Vue.js 3.0 Beta最新进展.我对直播的内容进行了一下整理.整整用了三天的空余时间赶上了 1. 全新文档RFCs Vue ...

  8. vue 3 学习笔记 (七)——vue3 中 computed 新用法

    vue3 中 的 computed 的使用,由于 vue3 兼容 vue2 的选项式API,所以可以直接使用 vue2的写法,这篇文章主要介绍 vue3 中 computed 的新用法,对比 vue2 ...

  9. Vue3.x+element-plus+ts踩坑笔记

    闲聊 前段时间小颖在B站找了个学习vue3+TS的视频,自己尝试着搭建了一些基础代码,在实现功能的过程中遇到了一些问题,为了防止自己遗忘,写个随笔记录一下嘻嘻 项目代码 git地址:vue3.x-ts ...

  10. vue3.0学习笔记(二)

    一.选择合适的ide 推荐使用vs code编辑器,界面清晰.使用方便,控制台功能很好用.webstorm也可以,看个人喜好. 二.ui框架选择 目前,pc端一般是选择element ui(饿了么), ...

随机推荐

  1. kvm使用桥接的方法

    什么是桥接 桥接就是把物理机的网卡模拟成交换机,虚拟机的网卡直接连在虚拟的网桥即交换机上.这样kvm虚拟机分配的IP地址,就应该和物理机在同一网段,可以对外进行服务. 在KVM下运行的VM默认的网卡采 ...

  2. 开启tcp_timestamps和tcp_tw_recycle造成NAT转发连接不上

    文章转载自:https://segmentfault.com/a/1190000022264813

  3. 基于CentOS 8服务器来搭建FastDFS高可用集群环境

    服务器版本 我们在服务器的命令行输入如下命令来查看服务器的内核版本. [root@localhost lib]# cat /etc/redhat-release CentOS Linux releas ...

  4. Spring mvc源码分析系列--Servlet的前世今生

    Spring mvc源码分析系列--Servlet的前世今生 概述 上一篇文章Spring mvc源码分析系列--前言挖了坑,但是由于最近需求繁忙,一直没有时间填坑.今天暂且来填一个小坑,这篇文章我们 ...

  5. Spring笔记三

    Spring-03 1. AOP 1.1 概念 ​ AOP为Aspect Oriented Programming的缩写,意为:面向切面编程.他是一种可以在不修改原来的核心代码的情况下给程序动态统一进 ...

  6. python的注释、变量

    注释 注释是代码的解释型语言,不会影响代码执行,就是专门给程序员看的. 注释是很重要的代码组成部分! # 单行注释 ''' 多行注释 连续输入三个单引号 ''' """ ...

  7. HYSBZ1036 [ZJOI2008]树的统计(树链剖分)

    将树通过树链剖分转化成线性序列,用线段树维护最值,和值即可. 1 #include <bits/stdc++.h> 2 using namespace std; 3 const int N ...

  8. GC plan_phase二叉树挂接的一个算法

    楔子 在看GC垃圾回收plan_phase的时候,发现了一段特殊的代码,仔细研究下得知,获取当前数字bit位里面为1的个数. 通过这个bit位为1的个数(count),来确定挂接当前二叉树子节点的一个 ...

  9. 网络安全(一)主动进攻之DNS基础和ettercap实现DNS流量劫持

    alittlemc,个人原创,个人理解和观点.若有错误.不理解请与我联系,谢谢! 介绍了DNS的解析过程. DNS劫持的思路和实践. DNS 域名 以为live.bilibili.com为例子,从后到 ...

  10. JUC(7)四大函数式接口

    文章目录 1.四大函数式接口(必须掌握) 1.1 function 1.2 Predicate 1.3 Consumer 1.4 Supplier 1.四大函数式接口(必须掌握) 1.lambda表达 ...