了解Object.defineProperty()

github源码

Object.defineProperty()方法直接在一个对象上定义一个新属性,或者修改一个已经存在的属性, 并返回这个对象。

vueJS采用 ES5 提供的 Object.defineProperty() 方法,监控对数据的操作,从而可以自动触发数据同步。并且,由于是在不同的数据上触发同步,可以精确的将变更发送给绑定的视图,而不是对所有的数据都执行一次检测。

首先我们得先知道,ECMAScript中有两种属性:数据属性和访问器属性( ie8以下只能在dom对象上使用;不能使用在普通对象上)

数据属性:
  [[Configurable]]: 表示能否修改属性。默认值为true

  [[Enumerable]]: 表示属性是否可枚举,也就是是否可以通过for-in循环返回属性。默认值为true

  [[Writable]]: 表示能否修改属性的值。默认值为true

  [[value]]: 包含这个属性的值.读取属性的时候就是通过这里开始读。默认值为undefined
访问器属性:
   [[Configurable]]: 表示能否修改属性。默认值为true

   [[Enumerable]]: 表示属性是否可枚举,也就是是否可以通过for-in循环返回属性。默认值为true

   [[Get]]: 在读取属性时调用的函数,默认时undefined

   [[Set]]: 在设置属性时调用的函数,默认时undefined

    我们要是想修改默认属性的值就可以使用:Object.defineProperty(obj,prop,descriptor);

1.基本用法:

var a= {}
Object.defineProperty(a,"b",{
value:123
});
console.log(a.b);//123

2.参数介绍:

 第一个参数obj:目标对象a

 第二个参数prop:需要定义的属性或方法的名字"b"

 第二个参数descriptor:目标属性所拥有的特性

2.1 第三个参数的取值介绍(descriptor)

    value:属性的值

    writable:如果为false,属性的值就不能被重写,只能为只读了

    configurable:总开关,一旦为false,就不能再设置他的(value,writable,configurable)

    enumerable:是否能在for...in循环中遍历出来或在Object.keys中列举出来。

    get:后面介绍

    set:后面介绍

    注意:在 descriptor 中不能同时设置访问器(get 和 set)和 wriable 或 value,否则会错,就是说用 get 和 set,就不能用 writable 或 value 中的任何一个

在基本用法里只设置了value,没有设置别的,可以简单的理解为(暂时这样理解)它会默认帮我们把writable,configurable,enumerable。都设上值,而且值还都是false。(仅限于第一次设置的时候),等同于以下代码:

    var a = {};
Object.defineProperty(a, 'b', {
value: 123,
writable: false,
enumerable: false,
configurable: false
});
console.log(a.b); //123

2.1.1 configurable介绍

总开关,第一次设置 false 之后,,第二次什么设置也不行了:
也就是说,你可以使用Object.defineProperty()方法无限修改同一个属性,但是当把configurable改为false之后就有限制了 var a = {};
Object.defineProperty(a, 'b', {
configurable: false
});
Object.defineProperty(a, 'b',{
configurable: true
});
//报错:Uncaught TypeError: Cannot redefine property: b(…)

2.1.2 writable介绍

var a = {};
Object.defineProperty(a, 'b', {
value: 123,
writable: false //只读
});
console.log(a.b); // 打印 123
a.b = 124; // 没有错误抛出(在严格模式下会抛出,即使之前已经有相同的值)
console.log(a.b); // 打印 123, 赋值不起作用。

2.1.3 enumerable介绍

var a = {}
Object.defineProperty(a,"b",{
value:3445,
enumerable:true
});
console.log(Object.keys(a));// 打印["b"] //改成false: var a = {}
Object.defineProperty(a,"b",{
value:3445,
enumerable:false
});
console.log(Object.keys(a));// 打印[]

2.1.4 set & get

访问器属性不能直接定义!只能通过Object.defineProperty()来定义:
var a= {}
Object.defineProperty(a,"b",{
set:function(newValue){
console.log("赋值是:"+newValue)
},
get:function(){
console.log("取值:")
return 2 //注意这里,我硬编码返回2
}
});
a.b =1; //赋值是: 1
console.log(a.b) ; //取值 2 简单来说,这个 b 赋值或者取值的时候会分别触发 set 和 get 对应的函数

3.Object.defineProperty示例:

//判断是不是对象
function isObj(obj){
var type = Object.prototype.toString.call(obj);
return type === '[object Object]';
} //执行函数:
function objFun(obj){
if(isObj(obj)){
new Observer(obj);
}
} function Observer(obj){
this.data = obj;
this.walk(obj);
} //监听事件函数:
Observer.prototype.walk = function(obj){
for(var k in obj){
def(obj,k,obj[k])
}
} function def(obj,k,val){
Object.defineProperty(obj,k,{
configurable:true,
enumerable:true,
get:function(){
console.log('get取值');
return val;
},
set:function(newVal){
if(val === newVal){
return;
}
val = newVal;
console.log('set设置值')
}
});
} //测试:
var obj = {a:111,b:222};
objFun(obj);
console.log(obj.a)//get取值 222
obj.a = 333;//set设置值
console.log(obj)

4.Object.defineProperty实现数据和视图的联动:

html:

<div>
Object.defineProperty实现数据和视图的联动: <br>
<span id="nickName"></span>
<div id="introduce"></div>
</div> js:(视图控制器) var userInfo = {};
Object.defineProperty(userInfo,'nickName',{
get:function(){
return document.getElementById('nickName').innerHTML;
},
set:function(nick){
document.getElementById('nickName').innerHTML = nick
}
});
Object.defineProperty(userInfo,'introduce',{
get:function(){
return document.getElementById('introduce').innerHTML;
},
set:function(introduce){
document.getElementById('introduce').innerHTML = introduce
}
});
//console.log(userInfo)
userInfo.nickName = '我是nickName';
userInfo.introduce = '我是introduce' 上面设置userInfo的nickName属性时会调用set方法,更新DOM节点的HTML

系列文章的目录:

Vue双向绑定的实现原理系列(一):Object.defineproperty
Vue双向绑定的实现原理系列(二):设计模式
Vue双向绑定的实现原理系列(三):监听器Observer和订阅者Watcher
Vue双向绑定的实现原理系列(四):补充指令解析器compile

Vue双向绑定的实现原理系列(一):Object.defineproperty的更多相关文章

  1. Vue双向绑定的实现原理系列(四):补充指令解析器compile

    补充指令解析器compile github源码 补充下HTML节点类型的知识: 元素节点 Node.ELEMENT_NODE(1) 属性节点 Node.ATTRIBUTE_NODE(2) 文本节点 N ...

  2. Vue双向绑定的实现原理系列(三):监听器Observer和订阅者Watcher

    监听器Observer和订阅者Watcher 实现简单版Vue的过程,主要实现{{}}.v-model和事件指令的功能 主要分为三个部分 github源码 1.数据监听器Observer,能够对数据对 ...

  3. 梳理vue双向绑定的实现原理

    Vue 采用数据劫持结合发布者-订阅者模式的方式来实现数据的响应式,通过Object.defineProperty来劫持数据的setter,getter,在数据变动时发布消息给订阅者,订阅者收到消息后 ...

  4. Vue双向绑定的实现原理及简单实现

    vue数据双向绑定原理   vue数据双向绑定是通过(数据劫持)+(发布者-订阅者模式)的方式来实现的,而所谓的数据劫持就是通过Object.defineProperty() 来实现的,所谓的Obje ...

  5. Vue双向绑定原理,教你一步一步实现双向绑定

    当今前端天下以 Angular.React.vue 三足鼎立的局面,你不选择一个阵营基本上无法立足于前端,甚至是两个或者三个阵营都要选择,大势所趋. 所以我们要时刻保持好奇心,拥抱变化,只有在不断的变 ...

  6. vue双向绑定、Proxy、defineproperty

    本文原链接:https://www.jianshu.com/p/2df6dcddb0d7 前言 双向绑定其实已经是一个老掉牙的问题了,只要涉及到MVVM框架就不得不谈的知识点,但它毕竟是Vue的三要素 ...

  7. Vue.js双向绑定的实现原理和模板引擎实现原理(##########################################)

    Vue.js双向绑定的实现原理 解析 神奇的 Object.defineProperty 这个方法了不起啊..vue.js和avalon.js 都是通过它实现双向绑定的..而且Object.obser ...

  8. vue双向绑定的原理及实现双向绑定MVVM源码分析

    vue双向绑定的原理及实现双向绑定MVVM源码分析 双向数据绑定的原理是:可以将对象的属性绑定到UI,具体的说,我们有一个对象,该对象有一个name属性,当我们给这个对象name属性赋新值的时候,新值 ...

  9. vue实现双向绑定的简单原理: defineProperty

    <!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8&quo ...

随机推荐

  1. 【原创】数据库基础之Sqlite

    官方:https://www.sqlite.org/index.html 简介 SQLite is a C-language library that implements a small, fast ...

  2. MVC4学习要点记四

    一.使用原生SQL使用EF的一个优点就是自动帮我们生成SQL,这在常规情况下很方便,但有些情况下用EF却不适合.另外还有些特别复杂的语句,利用EF很难生成.所以,EF提供一组方法用来执行原生的SQL. ...

  3. 【weixin】微信企业号和公众号区别和关系是什么?

    在移动互联网快速发展和智能手机普遍应用的时代环境下,随着微信平台应用不断扩大和微信用户的迅速增加,微信公众号运营也有了很大的发展,企业.机构和个人纷纷迈入微信公众号运营的行列.微信公众号就是在微信公众 ...

  4. 汉明码(hamming code)

    hamming code用于磁盘RAID 2中, 关于汉明码的讲解可以看这篇博文,介绍的很详细.最重要是最后的结论: 汉明码属于分组奇偶校验,P4P2P1=000,说明接收方生成的校验位和收到的校验位 ...

  5. MFC六大核心机制

    MFC六大核心机制概述 我们选择了C++,主要是因为它够艺术.够自由,使用它我们可以实现各种想法,而MFC将多种可灵活使用的功能封装起来,我们岂能忍受这种“黑盒”操作?于是研究分析MFC的核心机制成为 ...

  6. The last packet successfully received from the server was 39,900 milliseconds ago问题解决

    1,之前用Mysql或者mycat的时候都没有这个问题.后来改为haproxy+keepalived+mycat后出现这个问题 2,网上查了很多说法,我按照网上说的改了 datasource: url ...

  7. HTML的学习(注释)

    <!--charset 编码字符集--> <!--UTF-8 万国码 gb2312 中国标准第2312条 中文,韩文....大部分的亚裔语言(繁体字不支持) GBK 在上面的基础之上 ...

  8. shell与其他语言不同点

    1.定义变量时,变量名不加美元符号($,PHP语言中变量需要),如: your_name="w3cschool.cn" 注意,变量名和等号之间不能有空格,这可能和你熟悉的所有编程语 ...

  9. mysql计算QPS

    首先连接上mysql: $ mysql -h .x -P3306 -uusername -p123456 进入Mysql之后,查询general_log: mysql> SHOW VARIABL ...

  10. JAVA语言程序设计课后习题----第五单元解析(仅供参考)

    1 本题是水题,题目要求你求最大值.最小值,建议你用Arrays.sort函数进行排序,最大值.最小值就可以确定了 import java.util.Arrays; import java.util. ...