Vue中的数据代理

Object.defineProperty方法

这个方法是给一个对象添加属性用的,是es6的方法。

js 复制代码
<script>
    let person = {
        name:'kunkun',
        sex:'男',
    }
    Object.defineProperty(person,'age',{
        value:'18'
    })
    console.log(person);
</script>

运行打开控制台,可以看到如下结果:

可以发现上面的age属性的颜色比其它属性的颜色更淡,说明该属性是不参与枚举的,也就是不参与遍历。当我们用Object.keys(person)遍历时,不会出现age属性。如下图:

Q:想要该属性加入枚举该怎么做呢?

A:在配置项里加入enumerable属性并将其值设为true即可。

js 复制代码
    Object.defineProperty(person,'age',{
        value:'18',
        enumerable:true,//控制属性是否可以枚举,默认为false
    })

Q:想要该属性能够被修改该怎么做呢?

A:在配置项里加入writable属性并将其值设为true即可。

js 复制代码
    Object.defineProperty(person,'age',{
        value:'18',
        enumerable:true,//控制属性是否可以枚举,默认为false
        writable:true,//控制该属性是否可以被修改,默认为false
    })

Q:想要该属性能够被删除该怎么做呢?

A:在配置项里加入configurable属性并将其值设为true即可。

js 复制代码
    Object.defineProperty(person,'age',{
        value:'18',
        enumerable:true,//控制属性是否可以枚举,默认为false
        writable:true,//控制该属性是否可以被修改,默认为false
        configurable:true,//控制该属性是否可以被删除,默认为false
    })

有这样一个需求,我有一个number变量,然后person对象有一个age属性,且该属性的值是来源于number变量。用Object.defineProperty方法,当number值更改时,age属性自动更改,该怎么做呢?如下代码:

js 复制代码
<script>
    let number = 18
    let person = {
        name:'kunkun',
        sex:'男',
    }
    Object.defineProperty(person,'age',{
        get(){
            console.log('读取到的age属性值为:',number);
            return number
        },
        //当修改person的age属性时,getter函数会被调用,且会收到修改的值        
        set(value){
            console.log('修改了age属性值为:',value);
            number = value
        }
    })
    console.log(person)
</script>

结果如下:

理解数据代理

数据代理:通过一个对象代理对另一个对象中属性的操作(读/写),先上一个简单的例子:

js 复制代码
<script>
    //数据代理:通过一个对象代理对另一个对象中属性的操作(读/写)
    //通过obj2读取到obj的x并且能够修改x的值
    let obj = {x:100}
    let obj2 = {y:200}
    Object.defineProperty(obj2,'x',{
        get(){
            return obj.x
        },
        set(value){
            obj.x= value
        }
    })
</script>

在控制台进行一顿操作后,运行结果如图:

如上图所示我们通过obj2对象修改掉了obj对象中的x属性对应的值。
这就是一个简单的数据代理。

Vue中的数据代理

数据代理如下图图示,这里直接用了尚硅谷讲课的图,相关章节在这里

其实Vue中的_data就近似于我们在写Vue实例时配置的data属性(_data中还有很多属性,是Vue内部用来做数据处理等操作的相关属性),vm用_data来取到我们在data中设置的值,比如vm._data.name的值就和vm.name的值是一模一样的。vm就相当于是一个数据代理。Vue中的数据代理能够更加方便的操作data中的数据。

Vue中数据代理的基本原理:

  1. 通过Object.defineProperty()把data对象中所有的属性添加到vm上。
  2. 为每个添加到vm上的属性指定一个getter和setter函数。
  3. 在getter和setter内部去操作(读/写)data中对应的属性。
相关推荐
鹏多多6 分钟前
前端2025年终总结:借着AI做大做强再创辉煌
前端·javascript
WebGISer_白茶乌龙桃12 分钟前
Cesium实现“悬浮岛”式,三维立体的行政区划
javascript·vue.js·3d·web3·html5·webgl
小Tomkk15 分钟前
⭐️ StarRocks Web 使用介绍与实战指南
前端·ffmpeg
计算机学姐17 分钟前
基于SpringBoot的汽车租赁系统【个性化推荐算法+数据可视化统计】
java·vue.js·spring boot·后端·spring·汽车·推荐算法
不一样的少年_19 分钟前
产品催: 1 天优化 Vue 官网 SEO?我用这个插件半天搞定(不重构 Nuxt)
前端·javascript·vue.js
-dcr20 分钟前
50.智能体
前端·javascript·人工智能·ai·easyui
BingoGo22 分钟前
免费可商用商业级管理后台 CatchAdmin V5 正式发布 插件化与开发效率的全面提升
vue.js·后端·php
行者9630 分钟前
Flutter跨平台开发适配OpenHarmony:进度条组件的深度实践
开发语言·前端·flutter·harmonyos·鸿蒙
云和数据.ChenGuang31 分钟前
Uvicorn 是 **Python 生态中用于运行异步 Web 应用的 ASGI 服务器**
服务器·前端·人工智能·python·机器学习
IT_陈寒32 分钟前
SpringBoot 3.0实战:这5个新特性让你的开发效率提升50%
前端·人工智能·后端