Vue 子组件修改自身属性
所有的 prop 都使得其父子 prop 之间形成了一个单向下行绑定:父级 prop 的更新会向下流动到子组件中,但是反过来则不行。这样会防止从子组件意外变更父级组件的状态,从而导致你的应用的数据流向难以理解。
另外,每次父级组件发生变更时,子组件中所有的 prop 都将会刷新为最新的值。这意味着你不应该在一个子组件内部改变 prop。如果你这样做了,Vue 会在浏览器的控制台中发出警告。
尝试通过下面的方法直接修改属性值
1 | this.$props.modelValue = true; |
Vue 将提出警告
Set operation on key “modelValue” failed: target is readonly.
且修改该属性值并不能修改父组件对应的变量,仅仅适用于基础数据类型。
注意在 JavaScript 中对象和数组是通过引用传入的,所以对于一个数组或对象类型的 prop 来说,在子组件中改变这个对象或数组本身将会影响到父组件的状态,且 Vue 无法为此向你发出警告。作为一个通用规则,应该避免修改任何 prop,包括对象和数组,因为这种做法无视了单向数据绑定,且可能会导致意料之外的结果。
正确的修改方法是触发子组件事件,父组件监听该事件并执行修改父组件的函数,以 v-model
为例,v-model
本身就是传递给子组件的 modelValue
属性,然后监听 update:modelValue
事件来做到双向数据绑定,所以我们也可以通过该方法来在代码中修改 modelValue
1 | this.$emit("update:modelValue", !this.$props.modelValue); |
v-model
将自动监听 update:modelValue
事件并将 modelValue
修改为触发事件时传递的参数值(即 $emit 的第二个参数)。
需要注意的是,这种通过触发事件-监听事件的数据流模式并不能马上生效,如果使用以下代码
1 | this.$emit("update:modelValue", !this.$props.modelValue); |
会发现输出到控制台的仍然是 modelValue
原先的值,这是因为触发事件-监听事件的模式下属性值的修改需要时间,改用以下代码会发现输出正常:
1 | this.$emit("update:modelValue", !this.$props.modelValue); |
关键词:单向数据流