欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Vue.js雙向綁定實(shí)現(xiàn)原理詳解

 更新時(shí)間:2016年12月22日 09:37:34   作者:kidney  
這篇文章主要為大家詳細(xì)介紹了Vue.js雙向綁定實(shí)現(xiàn)原理,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下

Vue.js最核心的功能有兩個(gè),一是響應(yīng)式的數(shù)據(jù)綁定系統(tǒng),二是組件系統(tǒng)。本文僅探究幾乎所有Vue的開篇介紹都會(huì)提到的hello world雙向綁定是怎樣實(shí)現(xiàn)的。先講涉及的知識(shí)點(diǎn),再參考源碼,用盡可能少的代碼實(shí)現(xiàn)那個(gè)hello world開篇示例。

參考文章:http://www.dbjr.com.cn/article/100819.htm

一、訪問器屬性

訪問器屬性是對象中的一種特殊屬性,它不能直接在對象中設(shè)置,而必須通過defineProperty()方法單獨(dú)定義。

var obj = { };

// 為obj定義一個(gè)名為hello的訪問器屬性

Object.defineProperty(obj, "hello", {

get: function () {return sth},

set: function (val) {/* do sth */}

})

obj.hello // 可以像普通屬性一樣讀取訪問器屬性

訪問器屬性的"值"比較特殊,讀取或設(shè)置訪問器屬性的值,實(shí)際上是調(diào)用其內(nèi)部特性:get和set函數(shù)。

obj.hello // 讀取屬性,就是調(diào)用get函數(shù)并返回get函數(shù)的返回值

obj.hello = "abc" // 為屬性賦值,就是調(diào)用set函數(shù),賦值其實(shí)是傳參

get和set方法內(nèi)部的this都指向obj,這意味著get和set函數(shù)可以操作對象內(nèi)部的值。另外,訪問器屬性的會(huì)"覆蓋"同名的普通屬性,因?yàn)樵L問器屬性會(huì)被優(yōu)先訪問,與其同名的普通屬性則會(huì)被忽略(也就是所謂的被"劫持"了)。

二、極簡雙向綁定的實(shí)現(xiàn)

此例實(shí)現(xiàn)的效果是:隨文本框輸入文字的變化,span中會(huì)同步顯示相同的文字內(nèi)容;在js或控制臺(tái)顯式的修改obj.name的值,視圖會(huì)相應(yīng)更新。這樣就實(shí)現(xiàn)了model =>view以及view => model的雙向綁定,并且是響應(yīng)式的。

以上就是Vue實(shí)現(xiàn)雙向綁定的基本原理。

三、分解任務(wù)

上述示例僅僅是為了說明原理。我們最終要實(shí)現(xiàn)的是:

首先將該任務(wù)分成幾個(gè)子任務(wù):

   1、輸入框以及文本節(jié)點(diǎn)與data中的數(shù)據(jù)綁定

   2、輸入框內(nèi)容變化時(shí),data中的數(shù)據(jù)同步變化。即view => model的變化。

   3、data中的數(shù)據(jù)變化時(shí),文本節(jié)點(diǎn)的內(nèi)容同步變化。即model => view的變化。

要實(shí)現(xiàn)任務(wù)一,需要對DOM進(jìn)行編譯,這里有一個(gè)知識(shí)點(diǎn):DocumentFragment。

四、DocumentFragment

DocumentFragment(文檔片段)可以看作節(jié)點(diǎn)容器,它可以包含多個(gè)子節(jié)點(diǎn),當(dāng)我們將它插入到DOM中時(shí),只有它的子節(jié)點(diǎn)會(huì)插入目標(biāo)節(jié)點(diǎn),所以把它看作一組節(jié)點(diǎn)的容器。使用DocumentFragment處理節(jié)點(diǎn),速度和性能遠(yuǎn)遠(yuǎn)優(yōu)于直接操作DOM。Vue進(jìn)行編譯時(shí),就是將掛載目標(biāo)的所有子節(jié)點(diǎn)劫持(真的是劫持)到DocumentFragment中,經(jīng)過一番處理后,再將DocumentFragment整體返回插入掛載目標(biāo)。

五、數(shù)據(jù)初始化綁定

以上代碼實(shí)現(xiàn)了任務(wù)一,我們可以看到,hello world已經(jīng)呈現(xiàn)在輸入框和文本節(jié)點(diǎn)中。

六、響應(yīng)式的數(shù)據(jù)綁定

再來看任務(wù)二的實(shí)現(xiàn)思路:當(dāng)我們在輸入框輸入數(shù)據(jù)的時(shí)候,首先觸發(fā)input事件(或者keyup、change事件),在相應(yīng)的事件處理程序中,我們獲取輸入框的value并賦值給vm實(shí)例的text屬性。我們會(huì)利用defineProperty將data中的text劫持為vm的訪問器屬性,因此給vm.text賦值,就會(huì)觸發(fā)set方法。在set方法中主要做兩件事,第一是更新屬性的值,第二留到任務(wù)三再說。

任務(wù)二也就完成了,text屬性值會(huì)與輸入框的內(nèi)容同步變化:

七、訂閱/發(fā)布模式(subscribe&publish)

text屬性變化了,set方法觸發(fā)了,但是文本節(jié)點(diǎn)的內(nèi)容沒有變化。如何讓同樣綁定到text的文本節(jié)點(diǎn)也同步變化呢?這里又有一個(gè)知識(shí)點(diǎn):訂閱發(fā)布模式。

訂閱發(fā)布模式(又稱觀察者模式)定義了一種一對多的關(guān)系,讓多個(gè)觀察者同時(shí)監(jiān)聽某一個(gè)主題對象,這個(gè)主題對象的狀態(tài)發(fā)生改變時(shí)就會(huì)通知所有觀察者對象。

發(fā)布者發(fā)出通知 => 主題對象收到通知并推送給訂閱者 => 訂閱者執(zhí)行相應(yīng)操作

之前提到的,當(dāng)set方法觸發(fā)后做的第二件事就是作為發(fā)布者發(fā)出通知:“我是屬性text,我變了”。文本節(jié)點(diǎn)則是作為訂閱者,在收到消息后執(zhí)行相應(yīng)的更新操作。

八、雙向綁定的實(shí)現(xiàn)

回顧一下,每當(dāng)new一個(gè)Vue,主要做了兩件事:第一個(gè)是監(jiān)聽數(shù)據(jù):observe(data),第二個(gè)是編譯HTML:nodeToFragement(id)。

在監(jiān)聽數(shù)據(jù)的過程中,會(huì)為data中的每一個(gè)屬性生成一個(gè)主題對象dep。

在編譯HTML的過程中,會(huì)為每個(gè)與數(shù)據(jù)綁定相關(guān)的節(jié)點(diǎn)生成一個(gè)訂閱者watcher,watcher會(huì)將自己添加到相應(yīng)屬性的dep中。

我們已經(jīng)實(shí)現(xiàn):修改輸入框內(nèi)容 => 在事件回調(diào)函數(shù)中修改屬性值 => 觸發(fā)屬性的set方法。

接下來我們要實(shí)現(xiàn)的是:發(fā)出通知dep.notify() => 觸發(fā)訂閱者的update方法 => 更新視圖。

這里的關(guān)鍵邏輯是:如何將watcher添加到關(guān)聯(lián)屬性的dep中。

在編譯HTML過程中,為每個(gè)與data關(guān)聯(lián)的節(jié)點(diǎn)生成一個(gè)Watcher。Watcher函數(shù)中發(fā)生了什么呢?

首先,將自己賦給了一個(gè)全局變量Dep.target;

其次,執(zhí)行了update方法,進(jìn)而執(zhí)行了get方法,get的方法讀取了vm的訪問器屬性,從而觸發(fā)了訪問器屬性的get方法,get方法中將該watcher添加到了對應(yīng)訪問器屬性的dep中;

再次,獲取屬性的值,然后更新視圖。

最后,將Dep.target設(shè)為空。因?yàn)樗侨肿兞浚彩莣atcher與dep關(guān)聯(lián)的唯一橋梁,任何時(shí)刻都必須保證Dep.target只有一個(gè)值。

至此,hello world雙向綁定就基本實(shí)現(xiàn)了。文本內(nèi)容會(huì)隨輸入框內(nèi)容同步變化,在控制器中修改vm.text的值,會(huì)同步反映到文本內(nèi)容中(但輸入框還沒有綁定,讀者可以自己試試)。

完整代碼:https://github.com/bison1994/two-way-data-binding

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

相關(guān)文章

  • 使用vue控制元素顯示隱藏方式

    使用vue控制元素顯示隱藏方式

    這篇文章主要介紹了使用vue控制元素顯示隱藏方式,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • Vue引入Stylus知識(shí)點(diǎn)總結(jié)

    Vue引入Stylus知識(shí)點(diǎn)總結(jié)

    在本篇文章里小編給各位整理的是一篇關(guān)于Vue引入Stylus知識(shí)點(diǎn)總結(jié)內(nèi)容,有需要的朋友們可以學(xué)習(xí)參考下。
    2020-01-01
  • 基于vue cli重構(gòu)多頁面腳手架過程詳解

    基于vue cli重構(gòu)多頁面腳手架過程詳解

    本文分步驟給大家介紹了基于vue cli重構(gòu)多頁面腳手架過程,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友參考下
    2018-01-01
  • vue-router鉤子執(zhí)行順序示例解析

    vue-router鉤子執(zhí)行順序示例解析

    這篇文章主要為大家介紹了vue-router鉤子執(zhí)行順序示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2023-07-07
  • axios實(shí)現(xiàn)文件上傳并獲取進(jìn)度

    axios實(shí)現(xiàn)文件上傳并獲取進(jìn)度

    這篇文章主要為大家詳細(xì)介紹了axios實(shí)現(xiàn)文件上傳并獲取進(jìn)度,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2019-09-09
  • VUE3使用JSON編輯器方式

    VUE3使用JSON編輯器方式

    這篇文章主要介紹了VUE3使用JSON編輯器方式,具有很好的參考價(jià)值,希望對大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-10-10
  • 五分鐘理解keep?alive用法及原理

    五分鐘理解keep?alive用法及原理

    這篇文章主要為大家介紹了keep?alive用法及原理示例解析,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-07-07
  • vue3中setup語法糖下父子組件間傳遞數(shù)據(jù)的方式

    vue3中setup語法糖下父子組件間傳遞數(shù)據(jù)的方式

    Vue3中父組件指的是包含一個(gè)或多個(gè)子組件的組件,它們通過props和事件等方式來傳遞數(shù)據(jù)和通信,這篇文章主要介紹了vue3中setup語法糖下父子組件間傳遞數(shù)據(jù)的方式,需要的朋友可以參考下
    2023-06-06
  • Vue通過moment插件實(shí)現(xiàn)獲取當(dāng)前月的第一天和最后一天

    Vue通過moment插件實(shí)現(xiàn)獲取當(dāng)前月的第一天和最后一天

    這篇文章主要介紹了Vue 結(jié)合插件moment 實(shí)現(xiàn)獲取當(dāng)前月的第一天和最后一天,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧
    2023-10-10
  • vue使用ElementUI時(shí)導(dǎo)航欄默認(rèn)展開功能的實(shí)現(xiàn)

    vue使用ElementUI時(shí)導(dǎo)航欄默認(rèn)展開功能的實(shí)現(xiàn)

    這篇文章主要介紹了vue使用ElementUI時(shí)導(dǎo)航欄默認(rèn)展開功能的實(shí)現(xiàn),本文通過實(shí)例代碼給大家介紹的非常詳細(xì),具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2018-07-07

最新評論