成人午夜视频全免费观看高清-秋霞福利视频一区二区三区-国产精品久久久久电影小说-亚洲不卡区三一区三区一区

vue中怎么實現(xiàn)觀察者模式

vue中怎么實現(xiàn)觀察者模式,很多新手對此不是很清楚,為了幫助大家解決這個難題,下面小編將為大家詳細講解,有這方面需求的人可以來學習下,希望你能有所收獲。

成都創(chuàng)新互聯(lián)主要從事做網(wǎng)站、網(wǎng)站建設、網(wǎng)頁設計、企業(yè)做網(wǎng)站、公司建網(wǎng)站等業(yè)務。立足成都服務和龍,10余年網(wǎng)站建設經(jīng)驗,價格優(yōu)惠、服務專業(yè),歡迎來電咨詢建站服務:028-86922220

1 響應式原理

讓我們先從相應式原理開始。我們可以通過Object.defineProterty()來自定義Object的getter和setter 從而達到我們的目的。

代碼如下

function observe(value, cb) {
 Object.keys(value).forEach((key) => defineReactive(value, key, value[key] , cb))
}

function defineReactive (obj, key, val, cb) {
 Object.defineProperty(obj, key, {
  enumerable: true,
  configurable: true,
  get: ()=>{
   /*....依賴收集等....*/
   /*Github:https://github.com/answershuto*/
   return val
  },
  set:newVal=> {
   val = newVal;
   cb();/*訂閱者收到消息的回調(diào)*/
  }
 })
}

class Vue {
 constructor(options) {
  this._data = options.data;
  observe(this._data, options.render)
 }
}

let app = new Vue({
 el: '#app',
 data: {
  text: 'text',
  text2: 'text2'
 },
 render(){
  console.log("render");
 }
})

通過observe函數(shù)對app.data上的每一個key和value都設定getter和setter。當value改變的時候觸發(fā)setter,就會觸發(fā)render這個函數(shù)。響應式的目的就達成,如果是視圖更新的話我們通過監(jiān)聽dom的input事件來觸發(fā)數(shù)據(jù)更新
但是現(xiàn)在我們只有在改變vue._data.text的時候才會觸發(fā)他們的setter,但是我想偷懶,只改變vue.text就能觸發(fā)到setter怎么做呢?

我們使用代理的方法

_proxy.call(this, options.data);/*構造函數(shù)中*/

/*代理*/
function _proxy (data) {
 const that = this;
 Object.keys(data).forEach(key => {
  Object.defineProperty(that, key, {
   configurable: true,
   enumerable: true,
   get: function proxyGetter () {
    return that._data[key];
   },
   set: function proxySetter (val) {
    that._data[key] = val;
   }
  })
 });
}

依賴收集

讓我們再來看看下面的代碼

new Vue({
 template: 
  `<div>
   <span>text1:</span> {{text1}}
   <span>text2:</span> {{text2}}
  <div>`,
 data: {
  text1: 'text1',
  text2: 'text2',
  text3: 'text3'
 }
});

當你的text3變化的時候,實際上text3并沒有被渲染,但是也會觸發(fā)一次render函數(shù),這顯然是不對的。所以我們需要收集依賴。

我們只需要在初始化的時候渲染一遍,那所有渲染所依賴的數(shù)據(jù)都會被觸發(fā)getter,這時候我們只要把這個數(shù)據(jù)放到一個列表里就好啦!

我們先來認識一下Dep(dependencies)這個類,下圖是一個最簡單的Dep類。我們可以把他理解為發(fā)布者(這點很重要!?。?/p>

class Dep {
 constructor () {
  this.subs = [];
 }

 addSub (sub: Watcher) {
  this.subs.push(sub)
 }

 removeSub (sub: Watcher) {
  remove(this.subs, sub)
 }
 /*Github:https://github.com/answershuto*/
 notify () {
  // stabilize the subscriber list first
  const subs = this.subs.slice()
  for (let i = 0, l = subs.length; i < l; i++) {
   subs[i].update()
  }
 }
}
function remove (arr, item) {
 if (arr.length) {
  const index = arr.indexOf(item)
  if (index > -1) {
   return arr.splice(index, 1)
 }
}

我們每次觸發(fā)getter的時候,只要把觸發(fā)的對象放到dep.sub里面就好啦!
但是現(xiàn)在問題來了,我們用什么來裝這個觸發(fā)的'對象',也可以說式訂閱者呢?

我們使用Watcher這個類

class Watcher {
 constructor (vm, expOrFn, cb, options) {
  this.cb = cb;
  this.vm = vm;

  /*在這里將觀察者本身賦值給全局的target,只有被target標記過的才會進行依賴收集*/
  Dep.target = this;
  /*Github:https://github.com/answershuto*/
  /*觸發(fā)渲染操作進行依賴收集*/
  this.cb.call(this.vm);
 }

 update () {
  this.cb.call(this.vm);
 }
}

vm即是vue實例, expOrFn就是{{a+b}}里面的a+b, cb就是回調(diào)函數(shù)就是return a+b, options是一些配置項。

Vue在第一次渲染列表的時候如果碰到{{xxx}}這樣的表達式,就會new Watcher()。解析里面的函數(shù),然后把當前的watcher實例賦給Dep.target(Dep.target是全局的,一次性只能有一個存在,因為Vue一次只處理一個依賴)。然后執(zhí)行回調(diào)函數(shù)。(這里看似是執(zhí)行回調(diào)函數(shù)渲染,其實又觸發(fā)了一次getter,然后就會把當前的依賴添加到sub里去)

接下來開始依賴收集

class Vue {
 constructor(options) {
  this._data = options.data;
  observer(this._data, options.render);
  let watcher = new Watcher(this, );
 }
}

function defineReactive (obj, key, val, cb) {
 /*在閉包內(nèi)存儲一個Dep對象*/
 const dep = new Dep();

 Object.defineProperty(obj, key, {
  enumerable: true,
  configurable: true,
  get: ()=>{
   if (Dep.target) {
    /*Watcher對象存在全局的Dep.target中*/
    dep.addSub(Dep.target);
   }
  },
  set:newVal=> {
   /*只有之前addSub中的函數(shù)才會觸發(fā)*/
   dep.notify();
  }
 })
}

Dep.target = null; //防止依賴重復添加

這兒我們通過示例來講解

<template>
 <div>
 {{a+b}}
 </div>
 <div>
 {{a-c}}
 </div>
</template>

<script>
let app = new Vue( {
 data :{
  a: 1,
  b: 2,
  c: 3
 }
 })

我們編譯到{{a+b}},會去實例化一個對應的Watcher對象,Watcher的構造函數(shù)中有這么一句

this.cb.call(this.vm);this.cb指的是function(){return a+b};this.vm指的是這個vue對象,這樣就會觸發(fā)vue.a和vue.b的getter方法,a,b都有自己的dep對象,我們通過Dep.target將這個Watcher對象就加到dep的subs數(shù)組中去了,當我們變更a或者b是就會觸發(fā)setter,進而觸發(fā)subs數(shù)組中的update方法,視圖中的a+b就會更新

有個小知識點:我們新建一個屬性對象時必須通過Vue.set的方法去實現(xiàn),而不能直接通過=實現(xiàn),這樣會檢測不到,因為我們在初始化時就通過defineProperty重構了這個對象屬性的getter和setter方法,新建的屬性則沒有所以不會被檢測到

下圖為Vue框架在數(shù)據(jù)初始化中使用觀察者模式的示意圖:

vue中怎么實現(xiàn)觀察者模式

看完上述內(nèi)容是否對您有幫助呢?如果還想對相關知識有進一步的了解或閱讀更多相關文章,請關注創(chuàng)新互聯(lián)行業(yè)資訊頻道,感謝您對創(chuàng)新互聯(lián)的支持。

新聞名稱:vue中怎么實現(xiàn)觀察者模式
網(wǎng)站路徑:http://jinyejixie.com/article18/gpesdp.html

成都網(wǎng)站建設公司_創(chuàng)新互聯(lián),為您提供建站公司、小程序開發(fā)、做網(wǎng)站、動態(tài)網(wǎng)站、企業(yè)建站、標簽優(yōu)化

廣告

聲明:本網(wǎng)站發(fā)布的內(nèi)容(圖片、視頻和文字)以用戶投稿、用戶轉(zhuǎn)載內(nèi)容為主,如果涉及侵權請盡快告知,我們將會在第一時間刪除。文章觀點不代表本網(wǎng)站立場,如需處理請聯(lián)系客服。電話:028-86922220;郵箱:631063699@qq.com。內(nèi)容未經(jīng)允許不得轉(zhuǎn)載,或轉(zhuǎn)載時需注明來源: 創(chuàng)新互聯(lián)

網(wǎng)站托管運營
游戏| 马龙县| 祁连县| 扬中市| 敦化市| 邢台县| 和平区| 兴海县| 安乡县| 新巴尔虎右旗| 宁晋县| 拉萨市| 慈利县| 泽州县| 泾源县| 金乡县| 巍山| 高要市| 通州区| 武隆县| 南皮县| 吉首市| 吴忠市| 商南县| 乌兰浩特市| 突泉县| 武城县| 修武县| 太湖县| 临沂市| 德州市| 济宁市| 莒南县| 滕州市| 金寨县| 南汇区| 阿城市| 乃东县| 彭州市| 乌拉特后旗| 晋江市|