前言
在学习 Vue3 的时候,在项目实战中,需要用到 watch 监听 props 内属性的值的变化
但是却出现了无响应的现象
虽然让他可以响应只需要对 watch 的监听对象做一点小小的修改,但是我们还是需要研究一下具体为什么某些做法无法传值
...
props: {
testData: {
type: Object,
default: () => {}
}
}
......
setup(props) {
// 这种写法属于会有响应的情况
watch(
() => props.testData,
(newValue, oldValue) => {
console.log(newValue, 'tttttnewValue', oldValue, 'tttttoldValue')
}
)
// 这种写法属于不会有响应的情况
watch(
props.testData,
(newValue, oldValue) => {
console.log(newValue, 'tttttnewValue', oldValue, 'tttttoldValue')
}
)
}
问题分析
为了验证问题, 我进行了如下尝试
// test.vue
<template>
<my-form
:testData="test"
></my-form>
</template>
setup() {
const test = ref({})
const test1 = ref({})
watch(test, (newValue, oldValue) => {
let data = () => test
console.log(
isProxy(test),
isRef(test),
isProxy(test?.value),
isProxy(data()),
isRef(data()),
isReactive(test),
'test'
)
})
watch(test1, (newValue, oldValue) => {
let data = () => test1
console.log(
isProxy(test1),
isRef(test1),
isProxy(test1?.value),
isProxy(data()),
isRef(data()),
isReactive(test1),
'test1'
)
})
const getInfo = () => {
test.value = { ...formData.value }
test1.value = { ...formData.value }
}
const handleConfirmClick = () => {
getInfo()
// dialogVisible.value = false
}
return { test }
}
// MyForm.vue
....
props: {
testData: {
type: Object,
default: () => {}
}
}
...
setup(props) {
watch(
() => props.testData,
(newValue, oldValue) => {
let data = () => props.testData
console.log(
isProxy(props.testData),
isRef(props.testData),
isProxy(props.testData.value),
isProxy(data()),
isRef(data()),
isReactive(props.testData),
'testData'
)
console.log(newValue, 'tttttnewValue', oldValue, 'tttttoldValue')
}
)
}
打印出来的结果如下:
可以看出,在经过 prop,父传子操作之后,test 对象发生了一定量的改变
test 对象本身,在它所在的父组件中,依旧是一个 ref 对象,但是通过 prop 传值到子组件中后,获取 test 对象内值得对象却变成了一个 reactive 对象
然后就更加疑惑了,于是做出了下面的尝试
watch(props.testData, (newValue, oldValue) => {
let data = () => props.testData
console.log(
isProxy(props.testData),
isRef(props.testData),
isProxy(props.testData.value),
isProxy(data()),
isRef(data()),
isReactive(props.testData),
isReactive(props.testData.value),
'testData'
)
console.log(newValue, 'tttttnewValue', oldValue, 'tttttoldValue')
})
....
const getInfo = () => {
test.value.test = '11111'
}
结果监听到具体的打印了
于是乎开始了查看 Vue3 源码中 watch 的具体实现
// watch实现核心代码
let getter: () => any
let forceTrigger = false
if (isRef(source)) {
getter = () => {
console.log("getter重新执行了");
return (source as Ref).value;
}
forceTrigger = !!(source as Ref)._shallow
} else if (isReactive(source)) {
getter = () => source
deep = true
} else if (isArray(source)) {
getter = () =>
source.map(s => {
if (isRef(s)) {
return s.value
} else if (isReactive(s)) {
return traverse(s)
} else if (isFunction(s)) {
return callWithErrorHandling(s, instance, ErrorCodes.WATCH_GETTER, [
instance && (instance.proxy as any)
])
} else {
__DEV__ && warnInvalidSource(s)
}
})
} else if (isFunction(source)) {
if (cb) {
// getter with cb
getter = () =>
callWithErrorHandling(source, instance, ErrorCodes.WATCH_GETTER, [
instance && (instance.proxy as any)
])
} else {
// no cb -> simple effect
getter = () => {
if (instance && instance.isUnmounted) {
return
}
if (cleanup) {
cleanup()
}
return callWithAsyncErrorHandling(
source,
instance,
ErrorCodes.WATCH_CALLBACK,
[onInvalidate]
)
}
}
} else {
getter = NOOP
__DEV__ && warnInvalidSource(source)
}
然后我们就可以得出结论了
结论
// ref包裹一个对象时,返回值是一个ref对象,这个ref对象的value存储的一个指向proxy对象的引用
const test = ({})
// 当子应用内watch函数如下时
// (isReactive(source)) {
// getter = () => source
// deep = true
// }
// 在源码中, 监听的是一个proxy对象的时候,会直接把这个proxy对象转为getter函数
watch(props.testData, (newValue, oldValue) => {})
....
// 父应用内操作
const changeInfo = () => {
// 可以触发子组件watch的监听,因为修改的时proxy对象内的值,
// 当我们将ref对象传给子组件的时候,在模板中,ref对象会自动被解包,
// 也就时testData拿到的是test.value这个对象,也就是proxy对象。
// 子组件内监听到的就是这个proxy对象的变化,
// 而且watch函数自动对proxy函数监听设置为深度监听
test.value.name = '1'
}
// 父应用内操作
const changeInfo = () => {
// 无法触发子组件内的监听,因为我们直接修改了test的value,
// 相当于把test.value指向了另一个proxy对象。这里是value的指向发生了变化。
// 而子组件内监听的是testData内属性的变化,而不是test.value内存储的引用指向的变化
test.value = { ...item.value }
}
于是
// 当父组件内操作如下时
const changeInfo = () => {
test.value = { ...item.value }
}
// 我们在子组件内需要获取到 test.value 的变化,
// 也就是 引用地址的变化,props.testData这个变量,
//从根本上说是存储一个对象的引用地址的变量,而且是一个响应式数据,
//当数据发生变化时,和它有依赖的所有内容会做出响应。
//单纯从存储引用地址的响应式变量来说,它既不是ref对象,也不是proxy对象,
// 所以我们只能通过getter函数的形式来监听它的值的变化
watch(() => props.testData, (newValue, oldValue) => {})