国产成人精品久久免费动漫-国产成人精品天堂-国产成人精品区在线观看-国产成人精品日本-a级毛片无码免费真人-a级毛片毛片免费观看久潮喷

您的位置:首頁技術(shù)文章
文章詳情頁

Vue數(shù)據(jù)雙向綁定原理實例解析

瀏覽:84日期:2023-01-20 16:19:03

Vue數(shù)據(jù)雙向綁定原理是通過數(shù)據(jù)劫持結(jié)合發(fā)布者-訂閱者模式的方式來實現(xiàn)的,首先是對數(shù)據(jù)進(jìn)行監(jiān)聽,然后當(dāng)監(jiān)聽的屬性發(fā)生變化時則告訴訂閱者是否要更新,若更新就會執(zhí)行對應(yīng)的更新函數(shù)從而更新視圖

Vue數(shù)據(jù)雙向綁定原理實例解析

MVC模式

以往的MVC模式是單向綁定,即Model綁定到View,當(dāng)我們用JavaScript代碼更新Model時,View就會自動更新

Vue數(shù)據(jù)雙向綁定原理實例解析

MVVM模式

MVVM模式就是Model?View?ViewModel模式。它實現(xiàn)了View的變動,自動反映在 ViewModel,反之亦然。對于雙向綁定的理解,就是用戶更新了View,Model的數(shù)據(jù)也自動被更新了,這種情況就是雙向綁定。再說細(xì)點,就是在單向綁定的基礎(chǔ)上給可輸入元素input、textare等添加了change(input)事件,(change事件觸發(fā),View的狀態(tài)就被更新了)來動態(tài)修改model。

Vue數(shù)據(jù)雙向綁定原理實例解析

雙向綁定原理

vue數(shù)據(jù)雙向綁定是通過數(shù)據(jù)劫持結(jié)合發(fā)布者-訂閱者模式的方式來實現(xiàn)的

我們已經(jīng)知道實現(xiàn)數(shù)據(jù)的雙向綁定,首先要對數(shù)據(jù)進(jìn)行劫持監(jiān)聽,所以我們需要設(shè)置一個監(jiān)聽器Observer,用來監(jiān)聽所有屬性。如果屬性發(fā)上變化了,就需要告訴訂閱者Watcher看是否需要更新。因為訂閱者是有很多個,所以我們需要有一個消息訂閱器Dep來專門收集這些訂閱者,然后在監(jiān)聽器Observer和訂閱者Watcher之間進(jìn)行統(tǒng)一管理的。接著,我們還需要有一個指令解析器Compile,對每個節(jié)點元素進(jìn)行掃描和解析,將相關(guān)指令(如v-model,v-on)對應(yīng)初始化成一個訂閱者Watcher,并替換模板數(shù)據(jù)或者綁定相應(yīng)的函數(shù),此時當(dāng)訂閱者Watcher接收到相應(yīng)屬性的變化,就會執(zhí)行對應(yīng)的更新函數(shù),從而更新視圖。

因此接下去我們執(zhí)行以下3個步驟,實現(xiàn)數(shù)據(jù)的雙向綁定:

(1)實現(xiàn)一個監(jiān)聽器Observer,用來劫持并監(jiān)聽所有屬性,如果有變動的,就通知訂閱者。

(2)實現(xiàn)一個訂閱者Watcher,每一個Watcher都綁定一個更新函數(shù),watcher可以收到屬性的變化通知并執(zhí)行相應(yīng)的函數(shù),從而更新視圖。

(3)實現(xiàn)一個解析器Compile,可以掃描和解析每個節(jié)點的相關(guān)指令(v-model,v-on等指令),如果節(jié)點存在v-model,v-on等指令,則解析器Compile初始化這類節(jié)點的模板數(shù)據(jù),使之可以顯示在視圖上,然后初始化相應(yīng)的訂閱者(Watcher)。

Vue數(shù)據(jù)雙向綁定原理實例解析

實現(xiàn)一個Observer

Observer是一個數(shù)據(jù)監(jiān)聽器,其實現(xiàn)核心方法就是Object.defineProperty( )。如果要對所有屬性都進(jìn)行監(jiān)聽的話,那么可以通過遞歸方法遍歷所有屬性值,并對其進(jìn)行Object.defineProperty( )處理如下代碼實現(xiàn)了一個Observer。

function Observer(data) { this.data = data; this.walk(data);} Observer.prototype = { walk: function(data) { var self = this; //這里是通過對一個對象進(jìn)行遍歷,對這個對象的所有屬性都進(jìn)行監(jiān)聽 Object.keys(data).forEach(function(key) { self.defineReactive(data, key, data[key]); }); }, defineReactive: function(data, key, val) { var dep = new Dep(); // 遞歸遍歷所有子屬性 var childObj = observe(val); Object.defineProperty(data, key, { enumerable: true, configurable: true, get: function getter () { if (Dep.target) { // 在這里添加一個訂閱者 console.log(Dep.target) dep.addSub(Dep.target); }return val; }, // setter,如果對一個對象屬性值改變,就會觸發(fā)setter中的dep.notify(), 通知watcher(訂閱者)數(shù)據(jù)變更,執(zhí)行對應(yīng)訂閱者的更新函數(shù),來更新視圖。 set: function setter (newVal) { if (newVal === val) { return; } val = newVal;// 新的值是object的話,進(jìn)行監(jiān)聽 childObj = observe(newVal); dep.notify(); } }); }};function observe(value, vm) { if (!value || typeof value !== ’object’) { return; } return new Observer(value);};// 消息訂閱器Dep,訂閱器Dep主要負(fù)責(zé)收集訂閱者,然后在屬性變化的時候執(zhí)行對應(yīng)訂閱者的更新函數(shù)function Dep () { this.subs = [];}Dep.prototype = { /** * [訂閱器添加訂閱者] * @param {[Watcher]} sub [訂閱者] */ addSub: function(sub) { this.subs.push(sub); }, // 通知訂閱者數(shù)據(jù)變更 notify: function() { this.subs.forEach(function(sub) { sub.update(); }); }};Dep.target = null;

在Observer中,當(dāng)初我看別人的源碼時,我有一點不理解的地方就是Dep.target是從哪里來的,相信有些人和我會有同樣的疑問。這里不著急,當(dāng)寫到Watcher的時候,你就會發(fā)現(xiàn),這個Dep.target是來源于Watcher。

實現(xiàn)一個Watcher

Watcher就是一個訂閱者。用于將Observer發(fā)來的update消息處理,執(zhí)行Watcher綁定的更新函數(shù)。

如下代碼實現(xiàn)了一個Watcher

function Watcher(vm, exp, cb) { this.cb = cb; this.vm = vm; this.exp = exp; this.value = this.get(); // 將自己添加到訂閱器的操作} Watcher.prototype = { update: function() { this.run(); }, run: function() { var value = this.vm.data[this.exp]; var oldVal = this.value; if (value !== oldVal) { this.value = value; this.cb.call(this.vm, value, oldVal); } }, get: function() { Dep.target = this; // 緩存自己 var value = this.vm.data[this.exp] // 強(qiáng)制執(zhí)行監(jiān)聽器里的get函數(shù) Dep.target = null; // 釋放自己 return value; }};

在我研究代碼的過程中,我覺得最復(fù)雜的就是理解這些函數(shù)的參數(shù),后來在我輸出了這些參數(shù)之后,函數(shù)的這些功能也容易理解了。vm,就是之后要寫的SelfValue對象,相當(dāng)于Vue中的new Vue的一個對象。exp是node節(jié)點的v-model或v-on:click等指令的屬性值。

上面的代碼中就可以看出來,在Watcher的getter函數(shù)中,Dep.target指向了自己,也就是Watcher對象。在getter函數(shù)中,

var value = this.vm.data[this.exp] // 強(qiáng)制執(zhí)行監(jiān)聽器里的get函數(shù)。這里獲取vm.data[this.exp] 時,會調(diào)用Observer中Object.defineProperty中的get函數(shù)get: function getter () {if (Dep.target) { // 在這里添加一個訂閱者 console.log(Dep.target) dep.addSub(Dep.target);}return val; },

從而把watcher添加到了訂閱器中,也就解決了上面Dep.target是哪里來的這個問題。

實現(xiàn)一個Compile

Compile主要的作用是把new SelfVue 綁定的dom節(jié)點,(也就是el標(biāo)簽綁定的id)遍歷該節(jié)點的所有子節(jié)點,找出其中所有的v-指令和' {{}} '.

(1)如果子節(jié)點含有v-指令,即是元素節(jié)點,則對這個元素添加監(jiān)聽事件。(如果是v-on,則node.addEventListener(’click’),如果是v-model,則node.addEventListener(’input’))。接著初始化模板元素,創(chuàng)建一個Watcher綁定這個元素節(jié)點。

(2)如果子節(jié)點是文本節(jié)點,即' {{ data }} ',則用正則表達(dá)式取出' {{ data }} '中的data,然后var initText = this.vm[exp],用initText去替代其中的data。實現(xiàn)一個MVVM

可以說MVVM是Observer,Compile以及Watcher的“boss”了,他需要安排給Observer,Compile以及Watche做的事情如下

(1)Observer實現(xiàn)對MVVM自身model數(shù)據(jù)劫持,監(jiān)聽數(shù)據(jù)的屬性變更,并在變動時進(jìn)行notify

(2)Compile實現(xiàn)指令解析,初始化視圖,并訂閱數(shù)據(jù)變化,綁定好更新函數(shù)

(3)Watcher一方面接收Observer通過dep傳遞過來的數(shù)據(jù)變化,一方面通知Compile進(jìn)行view update。最后,把這個MVVM抽象出來,就是vue中Vue的構(gòu)造函數(shù)了,可以構(gòu)造出一個vue實例。最后寫一個html測試一下我們的功能

<!DOCTYPE html><html lang='en'><head> <meta charset='UTF-8'> <title>self-vue</title></head><style> #app { text-align: center; }</style><body> <div id='app'> <h2>{{title}}</h2> <input v-model='name'> <h1>{{name}}</h1> <button v-on:click='clickMe'>click me!</button> </div></body><script src='http://www.cgvv.com.cn/bcjs/js/observer.js'></script> <script src='http://www.cgvv.com.cn/bcjs/js/watcher.js'></script> <script src='http://www.cgvv.com.cn/bcjs/js/compile.js'></script> <script src='http://www.cgvv.com.cn/bcjs/js/mvvm.js'></script> <script type='text/javascript'> var app = new SelfVue({ el: ’#app’, data: { title: ’hello world’, name: ’canfoo’ }, methods: { clickMe: function () { this.title = ’hello world’; } }, mounted: function () { window.setTimeout(() => { this.title = ’你好’; }, 1000); } });</script></html>

先執(zhí)行mvvm中的new SelfVue(...),在mvvm.js中, 

observe(this.data);new Compile(options.el, this);

先初始化一個監(jiān)聽器Observer,用于監(jiān)聽該對象data屬性的值。

然后初始化一個解析器Compile,綁定這個節(jié)點,并解析其中的v-,' {{}} '指令,(每一個指令對應(yīng)一個Watcher)并初始化模板數(shù)

據(jù)以及初始化相應(yīng)的訂閱者,并把訂閱者添加到訂閱器中(Dep)。這樣就實現(xiàn)雙向綁定了。

如果v-model綁定的元素,

<input v-model='name'> 

即輸入框的值發(fā)生變化,就會觸發(fā)Compile中的

node.addEventListener(’input’, function(e) { var newValue = e.target.value; if (val === newValue) {return; } self.vm[exp] = newValue; val = newValue; });

self.vm[exp] = newValue;這個語句會觸發(fā)mvvm中SelfValue的setter,以及觸發(fā)Observer對該對象name屬性的監(jiān)聽,即Observer中的Object.defineProperty()中的setter。

setter中有通知訂閱者的函數(shù)dep.notify,Watcher收到通知后就會執(zhí)行綁定的更新函數(shù)。

最后的最后就是效果圖啦:

Vue數(shù)據(jù)雙向綁定原理實例解析

以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持好吧啦網(wǎng)。

標(biāo)簽: Vue
相關(guān)文章:
主站蜘蛛池模板: 亚洲视频在线精品 | 久久久久久久国产高清 | 综合久久99久久99播放 | 精品自拍视频在线观看 | 无码精品一区二区三区免费视频 | 亚洲三级视频 | 手机看片神马午夜 | 在线视频日韩 | 123成人网 | 亚洲欧美卡通成人制服动漫 | 欧洲一级片 | 精品国产乱码久久久久久一区二区 | 久久99久久精品国产只有 | 午夜成人在线视频 | 日本aa级片 | 国产大陆精品另类xxxx | 欧美精品免费线视频观看视频 | 福利云| 欧美午夜激情影院 | 国产精品免费视频能看 | 草草视频免费在线观看 | 视频一区中文字幕 | 亚洲一区二区三区在线 | 波多野结衣免费观看视频 | 色www永久免费网站国产 | 欧美午夜免费观看福利片 | 欧美成人性做爰网站免费 | 国产亚洲一级精品久久 | 日本美女黄色一级片 | 91欧美精品综合在线观看 | 毛片在线视频在线播放 | 亚洲精品午夜国产va久久成人 | 一区二区三区亚洲视频 | 国内精品小视频福利网址 | 一区二区高清在线 | 久久er热视频在这里精品 | 久久国产三级 | 成年大片免费视频播放二级 | 欧美在线视频免费 | 美国毛片免费一级 | 欧美区一区二区三 |