getter和setter
getter 是一種獲得屬性值的方法,setter是一種設置屬性值的方法。
屬性被賦值 a = 1的時候, a 的原型內的setter就會被觸發;
而 console.log(a) 的時候,a 的原型內的getter就會被觸發。
實現getter和setter
我們不能直接給變量的setter和getter 綁定事件函數,為了實現綁定我們要借助Object對象來構造帶有setter和getter的屬性。
這里有前輩總結的 幾種實現getter和setter的方法,而且他還總結了一些Object.prototype內控制屬性枚舉的特性的隱式屬性。
我這里選用了比較好構造的一種 Object.defineProperty
概要
Object.defineProperty() 方法直接在一個對象上定義一個新屬性,或者修改一個已經存在的屬性, 并返回這個對象。
語法
Object.defineProperty(obj, prop, descriptor)
參數
obj
需要定義屬性的對象。
prop
需被定義或修改的屬性名。
descriptor
需被定義或修改的屬性的描述符。
(function () {  var o = { a : 1}//聲明一個對象,包含一個 a 屬性,值為1  Object.defineProperty(o,"b",{    get: function () {      return this.a;    },    set : function (val) {      this.a = val;    },    configurable : true  });  console.log(o.b);//==> 1  o.b = 2;  console.log(o.b);//==> 2})();configurable是指 "b" 是否可以被再配置,默認是false。false的話
Object.defineProperty(o,"a",{set : function(val){}} );
再修改時會不起作用或者報錯,一般默認false。
構造我們的vue.watch
目標實現,以下是我們想要的達到的效果
import watcher from './watcher.js';let wm = new watcher({  data:{    a: 0   },  watch:{    a(newVal,oldVal){      console.log('newVal:'+newVal);      console.log('oldVal:'+oldVal);    }  }})vm.a = 1 // newVal:1// oldVal:0創建構造對象
class watcher{  constructor(opts){    this.$data = opts.data;    for(let key in opts.data){      this.setData(key,opts.data[key])    }  }  setData(_key,_val){    Object.defineProperty(this,_key,{      get: function () {        return this.$data[_key];      },      set : function (val) {        const oldVal = this.$data[_key];        if(oldVal === val)return val;        this.$data[_key] = val;        return val;      },    });  }}export default watcher;添加 watch事件觸發
/** * @desc 屬性改變監聽,屬性被set時出發watch的方法,類似vue的watch * @author Jason * @date 2018-04-27 * @constructor  * @param {object} opts - 構造參數. @default {data:{},watch:{}}; * @argument {object} data - 要綁定的屬性 * @argument {object} watch - 要監聽的屬性的回調  * watch @callback (newVal,oldVal) - 新值與舊值  */class watcher{  constructor(opts){    this.$data = this.getBaseType(opts.data) === 'Object' ? opts.data : {};    this.$watch = this.getBaseType(opts.watch) === 'Object' ? opts.watch : {};    for(let key in opts.data){      this.setData(key)    }  }  getBaseType(target) {    const typeStr = Object.prototype.toString.apply(target);      return typeStr.slice(8, -1);  }  setData(_key){    Object.defineProperty(this,_key,{      get: function () {        return this.$data[_key];      },      set : function (val) {        const oldVal = this.$data[_key];        if(oldVal === val)return val;        this.$data[_key] = val;        this.$watch[_key] && typeof this.$watch[_key] === 'function' && (          this.$watch[_key].call(this,val,oldVal)        );        return val;      },    });  }}export default watcher;            
新聞熱點
疑難解答
圖片精選