【一窥究竟系列】Wue实现数据双向绑定原理

一、认识Object.defineProperty()

语法:

Object.defineProperty(obj, prop, descriptor)

参数说明:

obj:必需。目标对象 
prop:必需。需定义或修改的属性的名字
descriptor:必需。目标属性所拥有的特性

返回值:

传入函数的对象。即第一个参数obj

给对象的属性添加特性描述,目前提供两种形式:数据描述和存取器描述。

A.数据描述

当修改或定义对象的某个属性的时候,给这个属性添加一些特性:

var obj = {
    test:"hello"
}
//对象已有的属性添加特性描述
Object.defineProperty(obj,"test",{
    configurable:true | false,
    enumerable:true | false,
    value:任意类型的值,
    writable:true | false
});
//对象新添加的属性的特性描述
Object.defineProperty(obj,"newKey",{
    configurable:true | false,
    enumerable:true | false,
    value:任意类型的值,
    writable:true | false
});

数据描述中的属性都是可选的,来看一下设置每一个属性的作用。

设置的特性总结:

value: 设置属性的值
writable: 值是否可以重写。true | false
enumerable: 目标属性是否可以被枚举。true | false
configurable: 目标属性是否可以被删除或是否可以再次修改特性 true | false

B.存取器描述

当使用存取器描述属性的特性的时候,允许设置以下特性属性:

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同上面的用法。

C.兼容性

在ie8下只能在DOM对象上使用,尝试在原生的对象使用 Object.defineProperty()会报错。

二、基本实例

简单的demo,很遗憾,暂未实现{{ message }}模板解析

<!DOCTYPE html>
<html lang="zh">
<head>
  <meta charset="UTF-8">
  <title>Wue 双向绑定实现原理</title>
</head>
<body>
<div id="app">
  <input type="text" w-model="message"/>
  <p>我会改变:<span id="message">message</span></p>
</div>
<script>
  // A.定义一个空对象, 获取被w-model设置的input
  let obj = {};
  let prop = document.querySelector('[w-model]').getAttribute('w-model');

  // 模板引擎函数
  let whtml = function (source, data) {
    let match;
    let reg = /{{(.+?)}}/g;
    while ((match = reg.exec(source)) !== null) {
      if (match[1]) {
        match[1] = match[1].replace(/\s/g, '');
        source = source.replace(match[0], data)
      } else {
        source = source.replace(match[0], '')
      }
    }
    return source
  };

  // B.精髓所在
  Object.defineProperty(obj, prop, {
    get: function () {
      return obj;
    },
    set: function (newValue) {
      document.querySelector('[w-model=message]').value = newValue;
      // let app = document.getElementById('app');
      // app.innerHTML = whtml(app.innerHTML, newValue);
      document.getElementById('message').innerHTML = newValue;
    }
  });

  // C.添加input事件监听
  document.addEventListener('input', function (e) {
    obj[prop] = e.target.value;
  });
</script>
</body>
</html>

猜你喜欢

转载自my.oschina.net/weijuer/blog/1796687
今日推荐