前言

在学习 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) => {})