国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 編程 > JavaScript > 正文

源碼分析Vue.js的監(jiān)聽實(shí)現(xiàn)教程

2019-11-19 16:45:43
字體:
供稿:網(wǎng)友

前言

相信一說到監(jiān)聽,當(dāng)然就離不了設(shè)計(jì)模式中鼎鼎大名的觀察者模式。舉個(gè)例子,你家后院著火了,可一定要等到煙霧很大火光很亮你才能發(fā)現(xiàn)啊,可是當(dāng)你安裝了一個(gè)火災(zāi)預(yù)警器,當(dāng)發(fā)生火災(zāi)就立馬能夠通知到你了。這就是一個(gè)典型的觀察者模式。當(dāng)然也還有一些其他變種,比如發(fā)布/訂閱(publish/subscribe)模式。

我們知道如果要將數(shù)據(jù)和視圖關(guān)聯(lián)起來,在數(shù)據(jù)變更的時(shí)候,同步視圖,同理視圖變更,數(shù)據(jù)也發(fā)生變化。vue.js是怎么實(shí)現(xiàn)這個(gè)的呢?下面我們來揭開它的神秘面紗。

demo:

<script src="../vue.js"> </script><div id="app"> <p> {{ message }} </p> <input v-model="message"></div><script type="text/javascript">new Vue({ el: '#app', data: { message: 'Hello Vue.js!' }});</script>set: function reactiveSetter(newVal) { var value = getter ? getter.call(obj) : val; if (newVal === value) { return; } if (setter) { setter.call(obj, newVal); } else { val = newVal; } childOb = observe(newVal); dep.notify();}

這段代碼出現(xiàn)在解析data屬性的時(shí)候,即調(diào)用Object.defineProperty方法配置data的屬性。一旦屬性發(fā)生變化,就notify發(fā)送廣播。

Dep.prototype.notify = function () { // stablize the subscriber list first var subs = toArray(this.subs); for (var i = 0, l = subs.length; i < l; i++) { subs[i].update(); }};

notify 最終是周知subscribe(訂閱者)更新,那么上面的數(shù)據(jù)變更就是發(fā)布者。 subscribe是Watcher這個(gè)類的實(shí)例化對(duì)象,在實(shí)例化的時(shí)候,會(huì)傳入回調(diào)函數(shù)來執(zhí)行update,vue弄了一個(gè)隊(duì)列來執(zhí)行watcher的更新函數(shù),具體可參考源碼。

Watcher.prototype.run = function () { …… if (value !== this.value || (isObject(value) || this.deep) && !this.shallow) { …… } else { this.cb.call(this.vm, value, oldValue); } } this.queued = this.shallow = false; } };

在Directive(指令)class中實(shí)例化了Watcher,_update函數(shù)負(fù)責(zé)來更新

var watcher = this._watcher = new Watcher(this.vm, this.expression, this._update, // callback { filters: this.filters, twoWay: this.twoWay, deep: this.deep, preProcess: preProcess, postProcess: postProcess, scope: this._scope });

在解析模板的時(shí)候會(huì)解析Directive,然后綁定,實(shí)例化watcher,這樣模板-data就關(guān)聯(lián)在一起了。

圖片描述

觀察者模式

林林總總的mvc或者mvvm框架基本也都是利用了觀察者模式,這個(gè)也非常有用,尤其在復(fù)雜的系統(tǒng)之中。

利用觀察者模式,在典型的ajax應(yīng)用中,回調(diào)的處理邏輯可以不跟請(qǐng)求耦合在一塊,這樣邏輯上也會(huì)更加清晰。如下是一個(gè)簡(jiǎn)單的發(fā)布/訂閱模式的實(shí)現(xiàn)

var PubSub = {};(function (q) { var topics = {}, subUid = -1; q.publish = function (topic) { if(!topics[topic]){  return false; } var subscribers = topics[topic],  len = subscribers ? subscribers.length : 0; while(len--){  var args = Array.prototype.slice.call(arguments, 1);  args.unshift(topic);  subscribers[len].callback.apply(this, args); } return this; }; q.subscribe = function (topic, callback) { if(!topics[topic]){  topics[topic] = []; } var subuid = (++subUid).toString(); topics[topic].push({  token: subuid,  callback: callback }); return subuid; }; q.unsubscribe = function (subid) { for(var k in topics){  if(topics[k]){  for(var i = 0, j = topics[k].length; i < j; i++){   if(topics[k][i].token === subid){   topics[k].splice(i, 1);   return subid;   }  }  } } return this; };})(PubSub);

這就是一個(gè)簡(jiǎn)單的訂閱發(fā)布系統(tǒng),每注冊(cè)一個(gè)訂閱者,其實(shí)就是將其回調(diào)處理的callback保存在一個(gè)字典對(duì)象的數(shù)組中,字典對(duì)象的key值可以隨意定義,只要與發(fā)布時(shí)的key對(duì)應(yīng)起來就好。

怎么使用呢?

<script>var messageLogger = function(){ console.log(JSON.stringify(arguments)); };var subscription = PubSub.subscribe('/newMessage', messageLogger);// {"0":"/newMessage","1":"hello world"}PubSub.publish('/newMessage', 'hello world');// {"0":"/newMessage","1":["test","a","b","c"]}PubSub.publish('/newMessage', ['test', 'a', 'b', 'c']);// {"0":"/newMessage","1":{"sender":"hello world","body":"hey man"}}PubSub.publish('/newMessage', { sender: 'hello world', body: 'hey man'});PubSub.unsubscribe(subscription);PubSub.publish('/newMessage', ['test', 'a', 'b', 'c'], 1);</script>

最后一個(gè)將不會(huì)打印出來,因?yàn)橐呀?jīng)取消訂閱了。

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對(duì)大家的學(xué)習(xí)或者工作能帶來一定的幫助,如果有疑問大家可以留言交流,謝謝大家對(duì)武林網(wǎng)的支持。

發(fā)表評(píng)論 共有條評(píng)論
用戶名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
主站蜘蛛池模板: 绥阳县| 海伦市| 贵溪市| 壶关县| 建阳市| 武胜县| 宁海县| 康马县| 镇平县| 榆中县| 新沂市| 罗甸县| 介休市| 闸北区| 罗山县| 镇雄县| 巧家县| 通江县| 茶陵县| 万荣县| 碌曲县| 桂平市| 台南县| 宾阳县| 巴林左旗| 隆德县| 上犹县| 双牌县| 商都县| 厦门市| 永福县| 青浦区| 平凉市| 察哈| 南丰县| 康马县| 临洮县| 波密县| 乌兰浩特市| 大厂| 浪卡子县|