当前位置: 首页 > article >正文

Object.defineProperty和响应式

Object.defineProperty()是一个监听对象属性变化的方法。一般情况下我们是不会直接使用的,或者说我们遇到的场景还没有这么高级。

最有名的例子就是Vue2的响应式实现,就是通过这个方法来实现的。

用起来不难,就是个API,只是用的比较少或者实现的功能都看起来很吊的样子,所以显得很高级。其实也就那样。

定义

Object.defineProperty() 静态方法会直接在一个对象上定义一个新属性,或修改其现有属性,并返回此对象。

定义新属性

运行下面的代码可以观察输出。

const obj1={}
Object.defineProperty(obj1,"property1",{
    value:1
})
console.log(obj1)//输出{}
console.log(obj1.property1)//输出1

obj1.property1输出1,这是符合我们的预期的。但obj1却输出{},我们期待的应该是{property1:1}。这就非常的奇怪了。明明obj1.property1可以输出property1的值,但输出obj1对象的时候却是空对象。

这就是Object.defineProperty()高级的一个地方。实际上,我们可以通过enumerable这个选项来控制属性是否对用户可见。

const obj1={}
Object.defineProperty(obj1,"property1",{
    value:1,
    enumerable:true
})
console.log(obj1)//输出{property1:1}
console.log(obj1.property1)//输出1

修改后,obj1的输出值就是{property1:1}。实际上我们利用defineProperty控制了属性的可见性。一般都会让用户可见。

数据描述符和访问器描述符

数据描述符指的是value和writable属性。访问器描述符指的get和set函数。两者是互斥的。

数据描述符

value的值就是属性的值。是可以读写的通过writable来控制的。

const obj1={}
Object.defineProperty(obj1,"property1",{
     value:1,
     writable:true,
})
obj1.property1=10
console.log(obj1.property1)//writable为false的时候,输出1,writable为true的时候,输出10,

writable为false的时候,obj1.property1=10并不会成功赋值。obj1.property1的值还是输出1,并不会报错。writable为true的时候,输出10,赋值成功。

访问器描述符

访问描述符是另一套读写数据的方式。不能够在get和set方法里面调用value,因为数据描述符和访问器描述符是互斥的。

const obj1 = {}
let property1Value = 1//必须自己定义一个变量来设置属性值
Object.defineProperty(obj1, "property1", {
    enumerable: true,
    get() {
        console.log("get:")
        return property1Value
    },
    set(newValue) {
        property1Value = newValue
    }
})
property1Value = 10
console.log(obj1)//输出{ property1: [Getter/Setter] }
console.log(obj1.property1)//输出10

在enumerable: true的情况下,obj1输出{ property1: [Getter/Setter] },就是长这样。enumerable: false的情况下,输出{}。

数据描述符和访问器描述符的主要区别

两者都是可以获取和修改属性值的。主要区别就是能不能监听到值的获取或者改变,也就是被Vue等控件玩出花的所谓响应式

1.不需要监听值的改变用数据描述符就行了,自带value值,没什么操作性。
2.需要监听值的改变就要用访问器描述符了。需要自己定义属性值的变量。读取的逻辑都需要自己实现,更灵活也更加强大复杂。


http://www.kler.cn/a/400385.html

相关文章:

  • 【LeetCode 题】只出现一次的数字--其余数字都出现3次
  • 递归(3)----力扣40组合数2,力扣473火柴拼正方形
  • Android Activity Manager Service (AMS) 深入详解及源码分析
  • STM32完全学习——使用标准库点亮LED
  • vs2022搭建opencv开发环境
  • 正则表达式语法详解(python)
  • 学习笔记025——Git基本基本命令
  • tcp 超时计时器
  • NLP论文速读(多伦多大学)|利用人类偏好校准来调整机器翻译的元指标
  • 华为OD机试-日志采集 E100
  • 线程(二)【线程控制】
  • YOLOv7-0.1部分代码阅读笔记-test.py
  • 使用Python编写一个简单的网站爬虫,从网站上抓取新闻标题和链接。
  • Bufferevent and SSL
  • 利用 `OpenCV` 和 `Matplotlib` 库进行图像读取、颜色空间转换、掩膜创建、颜色替换
  • Gin HTML 模板渲染
  • FPGA 第7讲 简单组合逻辑译码器
  • 案例精选 | 某知名教育集团基于安全运营平台的全域威胁溯源实践
  • 解决Ubuntu18.04及以上版本高分辨率下导致字体过小问题
  • linux开机不显示转到window
  • 鸿蒙中位置权限和相机权限
  • 远程jupyter lab的配置
  • H.265流媒体播放器EasyPlayer.js H.264/H.265播放器chrome无法访问更私有的地址是什么原因
  • ubuntu24.04设置开机自启动Eureka
  • VSCode 常用的快捷键
  • 使用WebHooks实现自动化工作流程的技术详解