日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区

您的位置:首頁技術文章
文章詳情頁

Vue數據雙向綁定原理實例解析

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

Vue數據雙向綁定原理是通過數據劫持結合發布者-訂閱者模式的方式來實現的,首先是對數據進行監聽,然后當監聽的屬性發生變化時則告訴訂閱者是否要更新,若更新就會執行對應的更新函數從而更新視圖

Vue數據雙向綁定原理實例解析

MVC模式

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

Vue數據雙向綁定原理實例解析

MVVM模式

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

Vue數據雙向綁定原理實例解析

雙向綁定原理

vue數據雙向綁定是通過數據劫持結合發布者-訂閱者模式的方式來實現的

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

因此接下去我們執行以下3個步驟,實現數據的雙向綁定:

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

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

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

Vue數據雙向綁定原理實例解析

實現一個Observer

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

function Observer(data) { this.data = data; this.walk(data);} Observer.prototype = { walk: function(data) { var self = this; //這里是通過對一個對象進行遍歷,對這個對象的所有屬性都進行監聽 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,如果對一個對象屬性值改變,就會觸發setter中的dep.notify(), 通知watcher(訂閱者)數據變更,執行對應訂閱者的更新函數,來更新視圖。 set: function setter (newVal) { if (newVal === val) { return; } val = newVal;// 新的值是object的話,進行監聽 childObj = observe(newVal); dep.notify(); } }); }};function observe(value, vm) { if (!value || typeof value !== ’object’) { return; } return new Observer(value);};// 消息訂閱器Dep,訂閱器Dep主要負責收集訂閱者,然后在屬性變化的時候執行對應訂閱者的更新函數function Dep () { this.subs = [];}Dep.prototype = { /** * [訂閱器添加訂閱者] * @param {[Watcher]} sub [訂閱者] */ addSub: function(sub) { this.subs.push(sub); }, // 通知訂閱者數據變更 notify: function() { this.subs.forEach(function(sub) { sub.update(); }); }};Dep.target = null;

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

實現一個Watcher

Watcher就是一個訂閱者。用于將Observer發來的update消息處理,執行Watcher綁定的更新函數。

如下代碼實現了一個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] // 強制執行監聽器里的get函數 Dep.target = null; // 釋放自己 return value; }};

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

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

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

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

實現一個Compile

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

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

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

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

(1)Observer實現對MVVM自身model數據劫持,監聽數據的屬性變更,并在變動時進行notify

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

(3)Watcher一方面接收Observer通過dep傳遞過來的數據變化,一方面通知Compile進行view update。最后,把這個MVVM抽象出來,就是vue中Vue的構造函數了,可以構造出一個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://m.b3g6.com/bcjs/js/observer.js'></script> <script src='http://m.b3g6.com/bcjs/js/watcher.js'></script> <script src='http://m.b3g6.com/bcjs/js/compile.js'></script> <script src='http://m.b3g6.com/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>

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

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

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

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

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

如果v-model綁定的元素,

<input v-model='name'> 

即輸入框的值發生變化,就會觸發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;這個語句會觸發mvvm中SelfValue的setter,以及觸發Observer對該對象name屬性的監聽,即Observer中的Object.defineProperty()中的setter。

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

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

Vue數據雙向綁定原理實例解析

以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持好吧啦網。

標簽: Vue
相關文章:
日本不卡不码高清免费观看,久久国产精品久久w女人spa,黄色aa久久,三上悠亚国产精品一区二区三区
国产不卡精品| 四虎精品一区二区免费| 日韩精品久久久久久久软件91| 亚洲经典在线| 美女91精品| 男女男精品网站| 男女男精品网站| 日韩欧美高清一区二区三区| 欧美色综合网| 国产精品国产三级在线观看| 麻豆91精品视频| 国产成人精品一区二区免费看京| 国产欧美日韩| 欧美国产一级| 91成人精品| 亚洲一二av| 91嫩草精品| 久久丁香四色| 免费污视频在线一区| 国产视频亚洲| 久久国产精品免费一区二区三区| 久久精品三级| 99久久精品国产亚洲精品| 亚洲少妇自拍| 久久精品97| 国产精品yjizz视频网| 美女亚洲一区| 蜜臀av一区二区在线免费观看| 日韩精品一页| 国产一区三区在线播放| 久久久久九九精品影院| 欧美交a欧美精品喷水| 午夜欧美视频| 国产一区福利| 亚洲制服欧美另类| 欧美精选视频一区二区| 日韩av电影一区| 国产主播一区| 欧美成人aaa| 亚洲+小说+欧美+激情+另类| 久久狠狠婷婷| 欧美国产视频| 性一交一乱一区二区洋洋av| 国产suv精品一区| 日韩高清中文字幕一区| 亚洲精品1区| 麻豆理论在线观看| 欧美专区一区| 亚洲免费一区二区| 中文字幕人成乱码在线观看| 日韩精品五月天| 欧美精选一区二区三区| jizzjizz中国精品麻豆| 国产日韩欧美三区| 亚洲精品极品| 久久国产精品99国产| 日本久久成人网| 久久免费精品| 欧美久久久网站| 在线看片日韩| 亚洲黄页一区| 激情综合激情| 精品成人免费一区二区在线播放| 国产精品多人| 国产亚洲精品美女久久| 天堂久久av| 蜜桃视频第一区免费观看| 在线日韩中文| 欧美精品高清| 在线看片国产福利你懂的| 美腿丝袜亚洲一区| 久久精品72免费观看| 亚洲日产av中文字幕| 一区在线免费| 九一成人免费视频| 久久精品国产大片免费观看| 日韩精品不卡一区二区| 97精品一区二区| 日韩av有码| 国产福利片在线观看| 成人午夜毛片| 精品久久影院| 精品视频网站| 精品三区视频| 麻豆91在线播放| 国产精品大片免费观看| 欧美亚洲综合视频| 日韩高清三区| 日韩高清三区| 日本视频一区二区| 国产一卡不卡| 国产精品第十页| 美女在线视频一区| 麻豆国产精品| 国产精品成久久久久| 福利视频一区| 日韩中文字幕高清在线观看| 亚洲成人不卡| 欧美日韩色图| 免费成人网www| 国产精品美女久久久| 性色一区二区| 蜜臀av一区二区在线免费观看| 中文字幕免费一区二区| 日本国产亚洲| 久久中文欧美| 日韩电影在线视频| 国产综合婷婷| 日韩一区欧美二区| 欧美亚洲免费| 久久精品国产99国产| 高清久久精品| 欧美二三四区| 日韩午夜黄色| 中文在线日韩| 欧美1区2区3| 欧美日韩精品免费观看视欧美高清免费大片 | 国产中文字幕一区二区三区| 国产一区二区三区不卡视频网站| 日韩电影二区| 五月天综合网站| 日韩一区欧美二区| 欧美视频一区| 国产一区二区三区亚洲| 日韩网站中文字幕| 亚洲欧美高清| 日韩中出av| 精品久久久久久久| 亚洲一级高清| 日韩精品亚洲专区| 国精品产品一区| 午夜国产欧美理论在线播放 | 7m精品国产导航在线| 精品黄色一级片| 激情视频一区二区三区| 日韩手机在线| 高清久久精品| 麻豆亚洲精品| 久久久亚洲欧洲日产| 久久高清精品| 亚洲欧美激情诱惑| 久久99偷拍| 最新亚洲激情| 欧美aaaaaa午夜精品| 欧美福利在线| 国产麻豆一区| 久久亚洲精品中文字幕蜜潮电影| 日韩在线网址| 欧美成人a交片免费看| 在线日韩成人| 福利精品一区| 久久亚洲视频| 国内一区二区三区| 日韩中文字幕91| 国产成人调教视频在线观看| 伊人久久婷婷| 精品国产一区二区三区噜噜噜| 国产字幕视频一区二区| 国产人成精品一区二区三| 99精品在线观看| 欧美欧美黄在线二区| 久久美女性网| 国产调教精品| 欧美女激情福利| 国产精品男女| 日韩视频中文| 国产在线观看91一区二区三区| 亚洲在线免费| 久久男人av资源站| 日本成人在线网站| 久久国产小视频| 国产精品v日韩精品v欧美精品网站| 米奇777超碰欧美日韩亚洲| 麻豆精品久久久| 亚洲精品第一| 日韩视频一区| 成人小电影网站| 国产欧美午夜| 美国三级日本三级久久99| 久久免费高清| 精品国产亚洲日本| 日韩在线观看中文字幕| 亚洲欧美综合| 亚洲黄色网址| 久久99蜜桃| 午夜天堂精品久久久久| 午夜日韩福利| 日韩欧美中文| 精品国产精品久久一区免费式| 日韩国产精品久久久| 亚洲在线观看| 在线日韩av| 成人欧美一区二区三区的电影| 欧美一级久久| 涩涩涩久久久成人精品| 99视频一区| 国产99久久| 成人精品中文字幕| 92国产精品| 欧美激情国产在线|