Vue实例里面的data属性为什么用函数返回

最近在学习Vue中在图灵社区买了一本电子书《Vue小书》,我感觉挺坑的,没有期待的那么好。

其中有的一下子就给一大串代码,但这一大串代码只是为了说明某一点,但是这片代码很多处都可以讲的,大概是因为篇幅限制原因吧。

其中我第一次看到的Vue实例中的data属性时,很好奇为什么是用function return 一个对象,而在之前某些视频中的写法又是直接写的对象,然后一查,又牵扯到JS原型链等一些问题。本质上是js语言的问题。

首先官方解释:

当一个组件被定义, data 必须声明为返回一个初始数据对象的函数,因为组件可能被用来创建多个实例。如果 data 仍然是一个纯粹的对象,则所有的实例将共享引用同一个数据对象!通过提供 data 函数,每次创建一个新实例后,我们能够调用 data 函数,从而返回初始数据的一个全新副本数据对象。
所以官方推荐的写法是:
data() {
return {
返回的数据
}
}

然后又举了一个例子:

假定存在组件ComponentA,里面定义为
module.exports = {
props: {
title: String
},
data: {
test: 1
}
}
在某个Page里面我们定义了如下的模板

<template>
    <div>
        <component-a title="1"></component-a>
        <component-a title="2"></component-a>
    </div>
</template>

那么上面两个ComponentA的实例中的data将同时为组件定义时data对应的对象,即相当于两个实例的data相互影响了。

也就是第一个实例的data和第二个我们并不想改变的实例的data也会发生改变。
再举个例子:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <div id="app">
        <my-btn></my-btn>
        <my-btn></my-btn>
        <my-btn></my-btn>
        <my-btn></my-btn>
        <my-btn></my-btn>
    </div>
    <!--祖册一个组件-->
    <template id='my_btn'>
        <button @click="counter += 1">点击的次数{{counter}}</button>
    </template>

    <script src="js/vue.min.js"></script>
    <script>
        Vue.component('my-btn', {
            template: '#my_btn',
            data() {
                return {
                    counter: 0
                }
            }
        })

        new Vue({
            el: '#app',
        })
    </script>
</body>
</html>
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
</head>
<body>
    <div id="app">
        <my-btn></my-btn>
        <my-btn></my-btn>
        <my-btn></my-btn>
        <my-btn></my-btn>
        <my-btn></my-btn>
    </div>
    <!--祖册一个组件-->
    <template id='my_btn'>
        <button @click="counter += 1">点击的次数{{counter}}</button>
    </template>

    <script src="js/vue.min.js"></script>
    <script>

        let data = {
            counter:0
        }

        Vue.component('my-btn', {
            template: '#my_btn',
            data() {
                return data;
            }
        })

        new Vue({
            el: '#app',
        })
    </script>
</body>
</html>

运行上面两段代码会发现不同结果。

这个问题类比到引用数据类型。
如果不用function return 每个组件的data都是内存的同一个地址,那一个数据改变其他也改变了,这当然就不是我们想要的。
用function return 其实就相当于申明了新的变量,相互独立,自然就不会有这样的问题

这让我之前看到的一个JS指针问题,js里面有引用类型和基本类型,

//基本类型
var a = 1;
var b = a;
b = 2;
console.log(a) //1 值不随b的改变而改变

//引用类型
var a = {x: 1};
var b = a;
b.x = 2
console.log(a.x) //2 值随着b的改变而改变

具体原因,犀牛书里面好像讲过, 基本类型占用空间少,而引用类型占用大,js为了节约空间,就把引用类型的值用一个叫做指针的东西和变量连接起来, 也就是说改一个其他的都会跟着改变,公用的意思.

我们用JS原型链在举一个例子:

var MyComponent = function() {}
MyComponent.prototype.data = {
  a: 1,
  b: 2,
}
// 上面是一个虚拟的组件构造器,真实的组件构造器方法很多

var component1 = new MyComponent()
var component2 = new MyComponent()
// 上面实例化出来两个组件实例,也就是通过<my-component>调用,创建的两个实例

component1.data.a === component2.data.a // true
component1.data.b = 5
component2.data.b // 5

可以看到上面代码中最后三句测试代码,如果两个实例同时引用一个对象,那么当你修改其中一个属性的时候,另外一个实例也会跟着改。这怎么可以,两个实例应该有自己各自的域才对。所以,需要通过下面方法来进行处理:

var MyComponent = function() {
  this.data = this.data()
}
MyComponent.prototype.data = function() {
  return {
    a: 1,
    b: 2,
  }
}

这个方法就是给实例添加一个data属性,它的值是原型方法data返回的一个新对象,下次用实例访问data的时候就会直接访问到这个对象,因为有了实例属性data了,就不会去查找原型上的data。

Vue.component('my-component', {
  template: '<div>测试</div>',
  data() {
    return {} // 返回一个唯一的对象,不要和其他组件共用一个对象进行返回
  },
})

上面这个操作是一个简易操作,实际上,它首先需要创建一个组件构造器,然后注册组件。注册组件的本质其实就是建立一个组件构造器的引用。使用组件才是真正创建一个组件实例。所以,注册组件其实并不产生新的组件类,但会产生一个可以用来实例化的新方式。

  • 3
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 在Vue中,data选项用于定义组件实例的初始数据。通常情况下,我们可以将data选项定义为一个对象,其中包含所有数据属性及其初始值。但是,有时候我们需要为每个组件实例提供一个独立的数据副本,以避免组件之间相互影响。为了实现这一点,我们可以使用一个函数来定义data选项。 使用函数式的data选项有两个好处: 1. 每个组件实例都会调用一次这个函数返回一个新的数据对象,从而避免了组件之间共享数据的问题。 2. 函数式的data选项可以访问组件实例属性,例如props和methods,从而可以动态地设置数据属性的初始值。 因此,使用函数式的data选项是一种更加灵活和安全的方式来定义组件的初始数据。 ### 回答2: 在Vue中,data属性通常用来定义组件的数据。通常情况下,我们可以直接将一个对象或者数组赋值给data属性。然而,在Vue中,推荐将data属性定义为一个返回一个对象的函数。 这是因为当我们将一个对象直接赋值给data属性时,这个对象会被所有组件实例共享,而且任何一个组件中对该对象的修改都会影响到其他组件。这是由于对象的引用传递机制导致的。 然而,如果我们将data属性定义为一个函数,每次创建一个组件实例时,都会调用一次该函数返回一个新的对象。这样,每个组件实例都有自己独立的数据对象,互不影响。 使用函数式的data可以确保每个组件实例都有独立的数据作用域,可以防止数据之间的混乱和错误修改。而如果直接将一个对象赋值给data属性,就会导致不同组件间的数据共享和潜在的数据错误。 因此,为了保证Vue组件的数据独立性和封装性,推荐将data属性定义为一个返回一个新对象的函数。这样每个组件实例都有自己独立的数据,避免了数据共享和潜在的数据错误。 ### 回答3: 在Vue中,data属性被用来定义组件的初始数据。为了保证每个组件实例data属性都是独立的,Vue规定data必须使用函数形式来定义,而不是直接使用对象。 具体来说,使用函数形式定义data属性有以下几个原因: 1. 避免数据共享:当使用对象形式定义data属性时,如果有多个组件实例共享同一个data对象,那么当其中一个组件实例修改了data的值时,其他组件实例也会受到影响,导致数据混乱。而通过函数形式定义data属性,每个组件实例都会返回一个新的data对象,确保每个实例之间数据是独立的,相互不会产生影响。 2. 提供数据复用:使用函数形式定义data属性使得我们可以在组件内部对数据进行定制化处理。通过在函数返回一个对象,我们可以对数据进行计算、筛选、过滤等操作,从而得到想要的数据结构。这样的数据复用有助于提高代码的可维护性和复用性。 3. 支持单文件组件:使用函数形式定义data属性也为单文件组件提供了便利。单文件组件是Vue中的一种组件组织方式,通过将组件的模板、样式和逻辑放在一个文件中来提高代码的可维护性。而使用函数形式定义data属性是单文件组件的一种约定,可以更方便地在模板和逻辑中使用data属性。 总结起来,Vue中使用函数形式定义data属性是为了保证每个组件实例的数据独立性,提供数据复用的能力,以及支持单文件组件的约定。这样可以有效地避免数据共享和产生数据混乱的问题,提高代码的可维护性和复用性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值