• <fieldset id="8imwq"><menu id="8imwq"></menu></fieldset>
  • <bdo id="8imwq"><input id="8imwq"></input></bdo>
    最新文章專題視頻專題問答1問答10問答100問答1000問答2000關(guān)鍵字專題1關(guān)鍵字專題50關(guān)鍵字專題500關(guān)鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關(guān)鍵字專題關(guān)鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
    問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
    當(dāng)前位置: 首頁 - 科技 - 知識百科 - 正文

    如何理解vue數(shù)據(jù)雙向綁定原理

    來源:懂視網(wǎng) 責(zé)編:小采 時間:2020-11-27 19:26:39
    文檔

    如何理解vue數(shù)據(jù)雙向綁定原理

    如何理解vue數(shù)據(jù)雙向綁定原理:Vue數(shù)據(jù)雙向綁定原理是通過數(shù)據(jù)劫持結(jié)合發(fā)布者-訂閱者模式的方式來實(shí)現(xiàn)的,首先是對數(shù)據(jù)進(jìn)行監(jiān)聽,然后當(dāng)監(jiān)聽的屬性發(fā)生變化時則告訴訂閱者是否要更新,若更新就會執(zhí)行對應(yīng)的更新函數(shù)從而更新視圖【推薦課程:Vue教程】MVC模式以往的MVC模式是單向綁定,即M
    推薦度:
    導(dǎo)讀如何理解vue數(shù)據(jù)雙向綁定原理:Vue數(shù)據(jù)雙向綁定原理是通過數(shù)據(jù)劫持結(jié)合發(fā)布者-訂閱者模式的方式來實(shí)現(xiàn)的,首先是對數(shù)據(jù)進(jìn)行監(jiān)聽,然后當(dāng)監(jiān)聽的屬性發(fā)生變化時則告訴訂閱者是否要更新,若更新就會執(zhí)行對應(yīng)的更新函數(shù)從而更新視圖【推薦課程:Vue教程】MVC模式以往的MVC模式是單向綁定,即M
    Vue數(shù)據(jù)雙向綁定原理是通過數(shù)據(jù)劫持結(jié)合發(fā)布者-訂閱者模式的方式來實(shí)現(xiàn)的,首先是對數(shù)據(jù)進(jìn)行監(jiān)聽,然后當(dāng)監(jiān)聽的屬性發(fā)生變化時則告訴訂閱者是否要更新,若更新就會執(zhí)行對應(yīng)的更新函數(shù)從而更新視圖

    【推薦課程:Vue教程】

    MVC模式

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

    MVVM模式

    MVVM模式就是Model–View–ViewModel模式。它實(shí)現(xiàn)了View的變動,自動反映在 ViewModel,反之亦然。對于雙向綁定的理解,就是用戶更新了View,Model的數(shù)據(jù)也自動被更新了,這種情況就是雙向綁定。再說細(xì)點(diǎn),就是在單向綁定的基礎(chǔ)上給可輸入元素input、textare等添加了change(input)事件,(change事件觸發(fā),View的狀態(tài)就被更新了)來動態(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è)置一個監(jiān)聽器Observer,用來監(jiān)聽所有屬性。如果屬性發(fā)上變化了,就需要告訴訂閱者Watcher看是否需要更新。因?yàn)橛嗛喺呤怯泻芏鄠€,所以我們需要有一個消息訂閱器Dep來專門收集這些訂閱者,然后在監(jiān)聽器Observer和訂閱者Watcher之間進(jìn)行統(tǒng)一管理的。接著,我們還需要有一個指令解析器Compile,對每個節(jié)點(diǎn)元素進(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個步驟,實(shí)現(xiàn)數(shù)據(jù)的雙向綁定:

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

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

    (3)實(shí)現(xiàn)一個解析器Compile,可以掃描和解析每個節(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)。

    未標(biāo)題-4.jpg

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

    Observer是一個數(shù)據(jù)監(jiān)聽器,其實(shí)現(xiàn)核心方法就是Object.defineProperty( )。如果要對所有屬性都進(jìn)行監(jiān)聽的話,那么可以通過遞歸方法遍歷所有屬性值,并對其進(jìn)行Object.defineProperty( )處理
    如下代碼實(shí)現(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)初我看別人的源碼時,我有一點(diǎn)不理解的地方就是Dep.target是從哪里來的,相信有些人和我會有同樣的疑問。這里不著急,當(dāng)寫到Watcher的時候,你就會發(fā)現(xiàn),這個Dep.target是來源于Watcher。

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

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

    如下代碼實(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é)點(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] 時,會調(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是哪里來的這個問題。

    實(shí)現(xiàn)一個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),則對這個元素添加監(jiān)聽事件。(如果是v-on,則node.addEventListener('click'),如果是v-model,則node.addEventListener('input'))。接著初始化模板元素,創(chuàng)建一個Watcher綁定這個元素節(jié)點(diǎn)。

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

    實(shí)現(xiàn)一個MVVM

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

    (1)Observer實(shí)現(xiàn)對MVVM自身model數(shù)據(jù)劫持,監(jiān)聽數(shù)據(jù)的屬性變更,并在變動時進(jìn)行notify
    (2)Compile實(shí)現(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實(shí)例。

    最后寫一個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é)點(diǎn),并解析其中的v-," {{}} "指令,(每一個指令對應(yīng)一個Watcher)并初始化模板數(shù)據(jù)以及初始化相應(yīng)的訂閱者,并把訂閱者添加到訂閱器中(Dep)。這樣就實(shí)現(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ù)。
    最后的最后就是效果圖啦:

    GIF.gif

    聲明:本網(wǎng)頁內(nèi)容旨在傳播知識,若有侵權(quán)等問題請及時與本網(wǎng)聯(lián)系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

    文檔

    如何理解vue數(shù)據(jù)雙向綁定原理

    如何理解vue數(shù)據(jù)雙向綁定原理:Vue數(shù)據(jù)雙向綁定原理是通過數(shù)據(jù)劫持結(jié)合發(fā)布者-訂閱者模式的方式來實(shí)現(xiàn)的,首先是對數(shù)據(jù)進(jìn)行監(jiān)聽,然后當(dāng)監(jiān)聽的屬性發(fā)生變化時則告訴訂閱者是否要更新,若更新就會執(zhí)行對應(yīng)的更新函數(shù)從而更新視圖【推薦課程:Vue教程】MVC模式以往的MVC模式是單向綁定,即M
    推薦度:
    標(biāo)簽: 如何 原理 VUE
    • 熱門焦點(diǎn)

    最新推薦

    猜你喜歡

    熱門推薦

    專題
    Top
    主站蜘蛛池模板: 亚洲国产精品碰碰| 国产999精品久久久久久| 欧美精品综合视频一区二区| 国产精品水嫩水嫩| 亚洲精品动漫免费二区| 国产成人精品久久亚洲高清不卡| 国产亚洲色婷婷久久99精品| 无码精品人妻一区二区三区免费| 日本人精品video黑人| 国产亚洲精品a在线无码| 一本久久精品一区二区| 国产综合精品蜜芽| 精品亚洲永久免费精品| 国产在线精品无码二区 | 久久亚洲中文字幕精品一区| 免费欧美精品a在线| 国产精品毛片无遮挡| 亚洲欧美精品午睡沙发| 欧美精品综合视频一区二区| 精品99又大又爽又硬少妇毛片| 四虎国产精品永久地址49| jizzjizz国产精品久久| 国产伦精品免编号公布| 久久夜色精品国产网站| 中文字幕日韩精品有码视频 | 精品久久久久中文字| 99爱在线视频这里只有精品| 九九精品在线观看| 精品久久久久久久| 久草视频在线这里精品| 久久精品国产精品青草app| 国产精品一级香蕉一区| 国产精品免费观看| 久久精品国产精品国产精品污 | 四虎永久在线精品免费一区二区 | 91不卡在线精品国产| 87国产私拍福利精品视频| AAA级久久久精品无码区| 99热这里只有精品国产66| 国产精品成人h片在线| 国产精品无码国模私拍视频|