Avoid mutating a prop directly since the value wi... 的错误处理

在vue.js中,在自定义一个组件的时候,可能会有这样的问题,当我们子组件中修改props中的值的时候,会有这样的错误提示:
Avoid mutating a prop directly since the value will be overwritten whenever the parent component re-renders. Instead, use a data or computed property based on the prop's value.
直译为:不要直接修改一个prop属性的值,因为这个值随时都可能在父组件重新渲染时被覆写。解决方案是,将prop属性的值赋给data或computed属性的值。
那么,具体我们该怎么做呢?首先,看一下vue.js的官方文档

官方文档

所有的 prop 都使得其父子 prop 之间形成了一个单向下行绑定:父级 prop 的更新会向下流动到子组件中,但是反过来则不行。这样会防止从子组件意外改变父级组件的状态,从而导致你的应用的数据流向难以理解。

额外的,每次父级组件发生更新时,子组件中所有的 prop 都将会刷新为最新的值。这意味着你不应该在一个子组件内部改变 prop。如果你这样做了,Vue 会在浏览器的控制台中发出警告(警告的内容就是上述那段英文)。

这里有两种常见的试图改变一个 prop 的情形:

这个 prop 用来传递一个初始值;这个子组件接下来希望将其作为一个本地的 prop 数据来使用。在这种情况下,最好定义一个本地的 data 属性并将这个 prop 用作其初始值:

props: ['initialCounter'],
data: function () {
  return {
    counter: this.initialCounter
  }
}

这个 prop 以一种原始的值传入且需要进行转换。在这种情况下,最好使用这个 prop 的值来定义一个计算属性:

props: ['size'],
computed: {
  normalizedSize: function () {
    return this.size.trim().toLowerCase()
  }
}

注意:在 JavaScript 中对象和数组是通过引用传入的,所以对于一个数组或对象类型的 prop 来说,在子组件中改变这个对象或数组本身将会影响到父组件的状态。

更进一步

官方文档中,有这样的一段描述:注意:在 JavaScript 中对象和数组是通过引用传入的,所以对于一个数组或对象类型的 prop 来说,在子组件中改变这个对象或数组本身将会影响到父组件的状态。意思是说:如果你的prop的值得类型不是js的基本类型(数值、字符串、null、undefined、布尔值),而是引用类型(对象、数组、函数、正则…)的话,就不能直接进行赋值了。那如何解决呢?
答案是:使用深拷贝(deepCopy)来代替直接赋值,具体什么是js的深拷贝,请自行查阅。同样是官方的例子,如果initialCounter是一个对象的话,可以这样赋值:

props: ['initialCounter'],
data: function () {
  return {
    counter: JSON.parse(JSON.stringify(this.initialCounter))
  }
}

但是这样'深拷贝'的方式有巨大的缺陷,只适用于一些简单的场景。想更深入的了解深拷贝,请自行查阅。

    原文作者:刘太刚
    原文地址: https://segmentfault.com/a/1190000019808519
    本文转自网络文章,转载此文章仅为分享知识,如有侵权,请联系博主进行删除。
点赞