亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

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

 更新時(shí)間:2020年05月15日 11:39:29   作者:程序猿的日常1  
這篇文章主要介紹了Vue數(shù)據(jù)雙向綁定原理實(shí)例解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下

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


MVC模式

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

MVVM模式

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

雙向綁定原理

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

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

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

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

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

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

實(shí)現(xiàn)一個(gè)Observer

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

function Observer(data) {  this.data = data;  this.walk(data);
}
 
Observer.prototype = {  walk: function(data) {    
var self = this;    //這里是通過對一個(gè)對象進(jìn)行遍歷,對這個(gè)對象的所有屬性都進(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) {         
 // 在這里添加一個(gè)訂閱者
 console.log(Dep.target)
 dep.addSub(Dep.target);
 }        return val;
 },     
 // setter,如果對一個(gè)對象屬性值改變,就會(huì)觸發(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é)收集訂閱者,然后在屬性變化的時(shí)候執(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)初我看別人的源碼時(shí),我有一點(diǎn)不理解的地方就是Dep.target是從哪里來的,相信有些人和我會(huì)有同樣的疑問。這里不著急,當(dāng)寫到Watcher的時(shí)候,你就會(huì)發(fā)現(xiàn),這個(gè)Dep.target是來源于Watcher。

實(shí)現(xiàn)一個(gè)Watcher

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

如下代碼實(shí)現(xiàn)了一個(gè)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的一個(gè)對象。exp是node節(jié)點(diǎn)的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] 時(shí),會(huì)調(diào)用Observer中Object.defineProperty中的get函數(shù)
get: function getter () {        
if (Dep.target) {         
// 在這里添加一個(gè)訂閱者         
console.log(Dep.target)          
dep.addSub(Dep.target);        
}        
return val;      
},

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

實(shí)現(xiàn)一個(gè)Compile

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

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

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

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

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

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

(3)Watcher一方面接收Observer通過dep傳遞過來的數(shù)據(jù)變化,一方面通知Compile進(jìn)行view update。
最后,把這個(gè)MVVM抽象出來,就是vue中Vue的構(gòu)造函數(shù)了,可以構(gòu)造出一個(gè)vue實(shí)例。最后寫一個(gè)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="js/observer.js"></script>
 <script src="js/watcher.js"></script>
 <script src="js/compile.js"></script>
 <script src="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);

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

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

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

如果v-model綁定的元素,

<input v-model="name"> 

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

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

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

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

相關(guān)文章

  • 基于Vue3實(shí)現(xiàn)掃碼槍掃碼并生成二維碼實(shí)例代碼

    基于Vue3實(shí)現(xiàn)掃碼槍掃碼并生成二維碼實(shí)例代碼

    vue3生成二維碼的方式有很多種,下面這篇文章主要給大家介紹了關(guān)于如何基于Vue3實(shí)現(xiàn)掃碼槍掃碼并生成二維碼的相關(guān)資料,文中通過實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下
    2023-06-06
  • vue-plugin-hiprint 詳細(xì)使用

    vue-plugin-hiprint 詳細(xì)使用

    這篇文章主要介紹了vue-plugin-hiprint 詳細(xì)使用說明,使用Vue.Draggable庫構(gòu)建可拖拽元素的示例,你可以根據(jù)具體需求和技術(shù)選型選擇適合的庫或方法來實(shí)現(xiàn)可拖拽元素的功能,需要的朋友可以參考下
    2023-08-08
  • 學(xué)習(xí)vue.js表單控件綁定操作

    學(xué)習(xí)vue.js表單控件綁定操作

    這篇文章主要和大家一起學(xué)習(xí)vue.js表單控件綁定操作,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-12-12
  • vue實(shí)現(xiàn)數(shù)字滾動(dòng)效果

    vue實(shí)現(xiàn)數(shù)字滾動(dòng)效果

    這篇文章主要為大家詳細(xì)介紹了vue實(shí)現(xiàn)數(shù)字滾動(dòng)效果,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2020-06-06
  • Vue Echarts實(shí)現(xiàn)圖表的動(dòng)態(tài)適配以及如何優(yōu)化

    Vue Echarts實(shí)現(xiàn)圖表的動(dòng)態(tài)適配以及如何優(yōu)化

    這篇文章主要介紹了Vue Echarts實(shí)現(xiàn)圖表的動(dòng)態(tài)適配以及如何優(yōu)化,在實(shí)際的前端開發(fā)過程中,動(dòng)態(tài)適配是一個(gè)非常重要的問題,在數(shù)據(jù)可視化的場景下,圖表的動(dòng)態(tài)適配尤為重要,需要的朋友可以參考下
    2023-05-05
  • vue點(diǎn)擊按鈕實(shí)現(xiàn)讓頁面的某一個(gè)元素全屏展示

    vue點(diǎn)擊按鈕實(shí)現(xiàn)讓頁面的某一個(gè)元素全屏展示

    這篇文章主要介紹了vue點(diǎn)擊按鈕實(shí)現(xiàn)讓頁面的某一個(gè)元素全屏展示,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-10-10
  • element?table數(shù)據(jù)量太大導(dǎo)致網(wǎng)頁卡死崩潰的解決辦法

    element?table數(shù)據(jù)量太大導(dǎo)致網(wǎng)頁卡死崩潰的解決辦法

    當(dāng)頁面數(shù)據(jù)過多,前端渲染大量的DOM時(shí),會(huì)造成頁面卡死問題,下面這篇文章主要給大家介紹了關(guān)于element?table數(shù)據(jù)量太大導(dǎo)致網(wǎng)頁卡死崩潰的解決辦法,需要的朋友可以參考下
    2023-02-02
  • 解決vuecli3中img src 的引入問題

    解決vuecli3中img src 的引入問題

    這篇文章主要介紹了解決vuecli3中img src 的引入問題,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-08-08
  • Vue?結(jié)合Sortablejs實(shí)現(xiàn)table行排序功能

    Vue?結(jié)合Sortablejs實(shí)現(xiàn)table行排序功能

    在一個(gè)列表展示頁面上,使用了表格組件,原有組件本身不支持拖拽功能,需求要求在列表的基礎(chǔ)上支持行拖拽排序,因此引入了www.sortablejs.com插件,接下來通過本文給大家講解Vue?結(jié)合Sortablejs實(shí)現(xiàn)table行排序功能,需要的朋友可以參考下
    2022-10-10
  • vue element table表格相同名稱列合并方式

    vue element table表格相同名稱列合并方式

    這篇文章主要介紹了vue element table表格相同名稱列合并方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-10-10

最新評論