日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

“约见”面试官系列之常见面试题第四十篇之双向绑定以及实现原理(建议收藏)

發(fā)布時間:2023/12/9 编程问答 32 豆豆
生活随笔 收集整理的這篇文章主要介紹了 “约见”面试官系列之常见面试题第四十篇之双向绑定以及实现原理(建议收藏) 小編覺得挺不錯的,現(xiàn)在分享給大家,幫大家做個參考.

目錄

?

MVC模式

MVVM模式

雙向綁定原理

1.實現(xiàn)一個Observer

2.實現(xiàn)一個Watcher

3.實現(xiàn)一個Compile

4.實現(xiàn)一個MVVM

最后寫一個html測試一下我們的功能


MVC模式

MVC模式

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

MVVM模式

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

?

MVVM模式

雙向綁定原理

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

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

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

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

1.實現(xiàn)一個Observer

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

function Observer(data) {this.data = data;this.walk(data);}Observer.prototype = {walk: function(data) {var self = this;//這里是通過對一個對象進行遍歷,對這個對象的所有屬性都進行監(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í)行對應訂閱者的更新函數(shù),來更新視圖。set: function setter (newVal) {if (newVal === val) {return;}val = newVal;// 新的值是object的話,進行監(jiān)聽childObj = observe(newVal);dep.notify();}});}};function observe(value, vm) {if (!value || typeof value !== 'object') {return;}return new Observer(value);};// 消息訂閱器Dep,訂閱器Dep主要負責收集訂閱者,然后在屬性變化的時候執(zhí)行對應訂閱者的更新函數(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中,當初我看別人的源碼時,我有一點不理解的地方就是Dep.target是從哪里來的,相信有些人和我會有同樣的疑問。這里不著急,當寫到Watcher的時候,你就會發(fā)現(xiàn),這個Dep.target是來源于Watcher。

2.實現(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] // 強制執(zhí)行監(jiān)聽器里的get函數(shù)Dep.target = null; // 釋放自己return value;}};

在我研究代碼的過程中,我覺得最復雜的就是理解這些函數(shù)的參數(shù),后來在我輸出了這些參數(shù)之后,函數(shù)的這些功能也容易理解了。vm,就是之后要寫的SelfValue對象,相當于Vue中的new Vue的一個對象。exp是node節(jié)點的v-model或v-on:click等指令的屬性值。如v-model="name",exp就是"name"。cb,就是Watcher綁定的更新函數(shù)。
上面的代碼中就可以看出來,在Watcher的getter函數(shù)中,Dep.target指向了自己,也就是Watcher對象。在getter函數(shù)中,

var value = this.vm.data[this.exp] // 強制執(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是哪里來的這個問題。

3.實現(xiàn)一個Compile

?

new SelfVue 綁定的dom節(jié)點


Compile主要的作用是把new SelfVue 綁定的dom節(jié)點,(也就是el標簽綁定的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 }} ",則用正則表達式取出" {{ data }} "中的data,然后var initText = this.vm[exp],用initText去替代其中的data。

里面有詳細的注釋。

4.實現(xiàn)一個MVVM

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

a、Observer實現(xiàn)對MVVM自身model數(shù)據(jù)劫持,監(jiān)聽數(shù)據(jù)的屬性變更,并在變動時進行notify
b、Compile實現(xiàn)指令解析,初始化視圖,并訂閱數(shù)據(jù)變化,綁定好更新函數(shù)
c、Watcher一方面接收Observer通過dep傳遞過來的數(shù)據(jù)變化,一方面通知Compile進行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="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);

先初始化一個監(jiān)聽器Observer,用于監(jiān)聽該對象data屬性的值。
然后初始化一個解析器Compile,綁定這個節(jié)點,并解析其中的v-," {{}} "指令,(每一個指令對應一個Watcher)并初始化模板數(shù)據(jù)以及初始化相應的訂閱者,并把訂閱者添加到訂閱器中(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ù)。
最后的最后就是效果圖啦:

本面試題為前端常考面試題,后續(xù)有機會繼續(xù)完善。我是歌謠,一個沉迷于故事的講述者。

歡迎一起私信交流。

“睡服“面試官系列之各系列目錄匯總(建議學習收藏)?

創(chuàng)作挑戰(zhàn)賽新人創(chuàng)作獎勵來咯,堅持創(chuàng)作打卡瓜分現(xiàn)金大獎

總結(jié)

以上是生活随笔為你收集整理的“约见”面试官系列之常见面试题第四十篇之双向绑定以及实现原理(建议收藏)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯,歡迎將生活随笔推薦給好友。