神奇的Object.defineProperty
vue.js和avalon.js 都是通过它实现双向绑定的。
对象是由多个名/值对组成的无序的集合。对象中每个属性对应任意类型的值。
定义对象可以使用构造函数或字面量的形式:
- var obj={};
- obj.name="玲"; //属性
- obj.run=function(){} //行为
还可以用Object.defineProperty定义新属性或修改原有的属性。
Object.defineProperty()
语法:
- Object.defineProperty(obj, prop, descriptor)
参数说明:
- obj:必需。目标对象
- prop:必需。需定义或修改的属性的名字
- descriptor:必需。目标属性所拥有的特性
返回值:
- 传入函数的对象。即第一个参数obj
针对属性,我们可以给这个属性设置一些特性,比如是否只读不可以写;是否可以被for..in或Object.keys()遍历。
给对象的属性添加特性描述,目前提供两种形式:数据描述和存取器描述。
数据描述:
当修改或定义对象的某个属性的时候,给这个属性添加一些特性:
- var obj = {
- test:hello world"
- }
- //对象已有的属性添加特性描述
- Object.defineProperty(obj,"test",{
- configurable:true | false,
- enumerable:true | false,
- value:任意类型的值,
- writable:true | false
- });
- //对象新添加的属性的特性描述
- Object.defineProperty(obj,"newtest",{
- configurable:true | false,
- enumerable:true | false,
- value:任意类型的值,
- writable:true | false
- });
设置的特性总结:
- value: 设置属性的值
- writable: 值是否可以重写。true | false
- enumerable: 目标属性是否可以被枚举。true | false
- configurable: 目标属性是否可以被删除或是否可以再次修改特性 true | false
每个属性的作用:
value
属性对应的值,可以使任意类型的值,默认为undefined
writable
属性的值是否可以被重写。设置为true可以被重写;设置为false,不能被重写。默认为false。
enumerable
此属性是否可以被枚举(使用for...in或Object.keys())。设置为true可以被枚举;设置为false,不能被枚举。默认为false。
configurable
是否可以删除目标属性或是否可以再次修改属性的特性(writable, configurable, enumerable)。设置为true可以被删除或可以重新设置特性;设置为false,不能被可以被删除或不可以重新设置特性。默认为false。
这个属性起到两个作用:
目标属性是否可以使用delete删除
目标属性是否可以再次设置特性
- //-----------------测试目标属性是否能被删除------------------------
- var obj = {}
- //第一种情况:configurable设置为false,不能被删除。
- Object.defineProperty(obj,"newKey",{
- value:"hello",
- writable:false,
- enumerable:false,
- configurable:false
- });
- //删除属性
- delete obj.newKey;
- console.log( obj.newKey ); //hello
- //第二种情况:configurable设置为true,可以被删除。
- Object.defineProperty(obj,"newKey",{
- value:"hello",
- writable:false,
- enumerable:false,
- configurable:true
- });
- //删除属性
- delete obj.newKey;
- console.log( obj.newKey ); //undefined
- //-----------------测试是否可以再次修改特性------------------------
- var obj = {}
- //第一种情况:configurable设置为false,不能再次修改特性。
- Object.defineProperty(obj,"newKey",{
- value:"hello",
- writable:false,
- enumerable:false,
- configurable:false
- });
- //重新修改特性
- Object.defineProperty(obj,"newKey",{
- value:"hello",
- writable:true,
- enumerable:true,
- configurable:true
- });
- console.log( obj.newKey ); //报错:Uncaught TypeError: Cannot redefine property: newKey
- //第二种情况:configurable设置为true,可以再次修改特性。
- Object.defineProperty(obj,"newKey",{
- value:"hello",
- writable:false,
- enumerable:false,
- configurable:true
- });
- //重新修改特性
- Object.defineProperty(obj,"newKey",{
- value:"hello",
- writable:true,
- enumerable:true,
- configurable:true
- });
- console.log( obj.newKey ); //hello
除了可以给新定义的属性设置特性,也可以给已有的属性设置特性
- //定义对象的时候添加的属性,是可删除、可重写、可枚举的。
- var obj = {
- test:"hello"
- }
- //改写值
- obj.test = 'change value';
- console.log( obj.test ); //'change value'
- Object.defineProperty(obj,"test",{
- writable:false
- })
- //再次改写值
- obj.test = 'change value again';
- console.log( obj.test ); //依然是:'change value'
一旦使用Object.defineProperty给对象添加属性,那么如果不设置属性的特性,那么configurable、enumerable、writable这些值都为默认的false
存取器描述
当使用存取器描述属性的特性的时候,允许设置以下特性属性:
- var obj = {};
- Object.defineProperty(obj,"newKey",{
- get:function (){} | undefined,
- set:function (value){} | undefined
- configurable: true | false
- enumerable: true | false
- });
注意:当使用了getter或setter方法,不允许使用writable和value这两个属性
getter/setter
当设置或获取对象的某个属性的值的时候,可以提供getter/setter方法。
getter 是一种获得属性值的方法
setter是一种设置属性值的方法。
在特性中使用get/set属性来定义对应的方法。
- var obj = {};
- var initValue = 'hello';
- Object.defineProperty(obj,"newKey",{
- get:function (){
- //当获取值的时候触发的函数
- return initValue;
- },
- set:function (value){
- //当设置值的时候触发的函数,设置的新值通过参数value拿到
- initValue = value;
- }
- });
- //获取值
- console.log( obj.newKey ); //hello
- //设置值
- obj.newKey = 'change value';
- console.log( obj.newKey ); //change value
注意:get或set不是必须成对出现,任写其一就可以。如果不设置方法,则get和set的默认值为undefined
configurable和enumerable同上面的用法。
兼容性
在ie8下只能在DOM对象上使用,尝试在原生的对象使用 Object.defineProperty()会报错。
神奇的Object.defineProperty的更多相关文章
- 解析神奇的 Object.defineProperty
这个方法了不起啊..vue.js是通过它实现双向绑定的..而且Object.observe也被草案发起人撤回了..所以defineProperty更有必要了解一下了. 几行代码看他怎么用 var a= ...
- 解析 神奇的 Object.defineProperty
这个方法了不起啊..vue.js和avalon.js 都是通过它实现双向绑定的..而且Object.observe也被草案发起人撤回了..所以defineProperty更有必要了解一下了几行代码看他 ...
- 神奇的 Object.defineProperty 解释说明
原文 : https://segmentfault.com/a/1190000004346467?utm_source=tuicool&utm_medium=referral 这个方法了不起啊 ...
- vue原理之-神奇的Object.defineProperty
vue2.0通过defineProperty进行数据双向绑定 例如:(他接受三个参数,都是必填!) var a= {} Object.defineProperty(a,"b",{ ...
- Vue.js 源码学习笔记 -- 分析前准备2 -- Object.defineProperty
解析神奇的 Object.defineProperty 几行代码看他怎么用 var a= {} Object.defineProperty( a, "b", { value ...
- 双向数据绑定实现之Object.defineProperty
vue.js利用的是es5的 defineproperty 特性实现的双向数据绑定,了解一下基本原理. 举例 var person= {}; Object.defineProperty(person, ...
- javascript之Object.defineProperty的奥妙
直切主题 今天遇到一个这样的功能: 写一个函数,该函数传递两个参数,第一个参数为返回对象的总数据量,第二个参数为初始化对象的数据.如: var o = obj (4, {name: 'xu', age ...
- Object.defineproperty实现数据和视图的联动
Object.defineproperty语法 var o = {}; // 创建一个新对象 // Example of an object property added with definePro ...
- Vue 双向数据绑定原理分析 以及 Object.defineproperty语法
第三方精简版实现 https://github.com/luobotang/simply-vue Object.defineProperty 学习,打开控制台分别输入以下内容调试结果 userInfo ...
随机推荐
- Ubuntu 16.04下Redis Cluster集群搭建(官方原始方案)
前提:先安装好Redis,参考:http://www.cnblogs.com/EasonJim/p/7599941.html 说明:Redis Cluster集群模式可以做到动态增加节点和下线节点,使 ...
- symfony 参考
多语言存数据库 http://blog.elendev.com/development/php/symfony/use-a-database-as-translation-provider-in-sy ...
- linux系统编程:线程同步-信号量(semaphore)
线程同步-信号量(semaphore) 生产者与消费者问题再思考 在实际生活中,仅仅要有商品.消费者就能够消费,这没问题. 但生产者的生产并非无限的.比如,仓库是有限的,原材料是有限的,生产指标受消费 ...
- frameset怎样实现整个页面的跳转
登录页面login.jsp,系统登录成功后展示mainLayout.jsp, 我如今用frameset框架把页面mainLayout.jsp分为三部分,head.jsp..left.jsp.right ...
- C#中的函数式编程:递归与纯函数(二) 学习ASP.NET Core Razor 编程系列四——Asp.Net Core Razor列表模板页面
C#中的函数式编程:递归与纯函数(二) 在序言中,我们提到函数式编程的两大特征:无副作用.函数是第一公民.现在,我们先来深入第一个特征:无副作用. 无副作用是通过引用透明(Referential ...
- SP2-0734: 未知的命令开头 "imp scott/..." - 忽略了剩余的行。
Oracle数据导入报错:SP2-0734: 未知的命令开头 "imp scott/..." - 忽略了剩余的行. 原因:进入sqlplus里是不能运行imp的(sqlplus不认 ...
- @Override用在哪儿
帮朋友改一段代码,看到好多红叉都是指向@Override. 是这样,他代码里写了一个接口.方法都用抽象函数声明在接口类里.然后在继承自这个接口的实现类里写详细方法的空壳 ...
- 图片存储系统TFS
1 TFS和GFS比较 1.1 GFS的应用场景 第一,百万级别的文件,并且是大文件,文件都是100MB以上,1G级别的文件很常见. 第二,集群是建立在商业计算机之上,并不可靠,监控各个节点的状态,当 ...
- mysql对表列数和行大小的限制
mysql对表列数和行大小的限制 - CSDN博客 https://blog.csdn.net/Dr_Joseph/article/details/78111312
- Codeforces Round #100 A. New Year Table
A. New Year Table time limit per test 2 seconds memory limit per test 256 megabytes input standard i ...