vue響應(yīng)式原理與雙向數(shù)據(jù)的深入解析
了解object.defineProperty 實(shí)現(xiàn)響應(yīng)式
清楚 observe/watcher/dep 具體指的是什么
了解 發(fā)布訂閱模式 以及其解決的具體問(wèn)題
在Javascript里實(shí)現(xiàn)數(shù)據(jù)響應(yīng)式一般有倆種方案,分別對(duì)應(yīng)著vue2.x 和 vue3.x使用的方式,他們分別是:
對(duì)象屬性攔截 (vue2.x) Object.defineProperty
對(duì)象整體代理 (vue3.x) Proxy
提示:以下是本篇文章正文內(nèi)容,下面案例可供參考
vue-響應(yīng)式是什么?
Vue 最獨(dú)特的特性之一,是其非侵入性的響應(yīng)式系統(tǒng)。數(shù)據(jù)模型僅僅是普通的 JavaScript對(duì)象。而當(dāng)你修改它們時(shí),視圖會(huì)進(jìn)行更新。這使得狀態(tài)管理非常簡(jiǎn)單直接,不過(guò)理解其工作原理同樣重要,這樣你可以避開(kāi)一些常見(jiàn)的問(wèn)題。在這個(gè)章節(jié),我們將研究一下
Vue 響應(yīng)式系統(tǒng)的底層的細(xì)節(jié)。
vue-響應(yīng)式如何實(shí)現(xiàn)的?
數(shù)據(jù)響應(yīng)式:數(shù)據(jù)模型僅僅是普通的JavaScript對(duì)象,而當(dāng)我們修改數(shù)據(jù)時(shí),視圖會(huì)進(jìn)行更新,避免了頻繁的DOM操作,提高開(kāi)發(fā)效率,這與Jquery不一樣,Jquery是頻繁的操作Dom
對(duì)雙向數(shù)據(jù)綁定的理解
數(shù)據(jù)改變,視圖改變,視圖改變,數(shù)據(jù)也隨之改變( 通過(guò)這句話(huà),我們可以看到在雙向綁定中是包含了數(shù)據(jù)響應(yīng)式的內(nèi)容)
我們可以使用v-model 在表單元素上創(chuàng)建雙向數(shù)據(jù)綁定
數(shù)據(jù)驅(qū)動(dòng)是Vue最獨(dú)特的特性之一
開(kāi)發(fā)過(guò)程中僅僅需要關(guān)注數(shù)據(jù)本身,不需要關(guān)心數(shù)據(jù)是如何渲染到視圖中的。主流的MVVM框架都已經(jīng)實(shí)現(xiàn)了數(shù)據(jù)響應(yīng)式與雙向綁定,所以可以將數(shù)據(jù)綁定到DOM上。
在vue.js中,所謂的數(shù)據(jù)驅(qū)動(dòng)就是當(dāng)數(shù)據(jù)發(fā)生變化的時(shí)候,用戶(hù)界面發(fā)生相應(yīng)的變化,開(kāi)發(fā)者不需要手動(dòng)的去修改dom。
對(duì)數(shù)據(jù)驅(qū)動(dòng)的理解:
那么vuejs是如何實(shí)現(xiàn)這種數(shù)據(jù)驅(qū)動(dòng)的呢?
vue實(shí)現(xiàn)數(shù)據(jù)雙向綁定主要是:采用數(shù)據(jù)劫持結(jié)合發(fā)布者-訂閱者模式的方式,通過(guò)Object.defineProperty()來(lái)劫持各個(gè)屬性的setter,getter,在數(shù)據(jù)變動(dòng)時(shí)發(fā)布消息給訂閱者,觸發(fā)相應(yīng)監(jiān)聽(tīng)回調(diào)。當(dāng)把一個(gè)普通Javascript 對(duì)象傳給 Vue 實(shí)例來(lái)作為它的 data 選項(xiàng)時(shí),Vue 將遍歷它的屬性,用Object.defineProperty 將它們轉(zhuǎn)為 getter/setter。用戶(hù)看不到 getter/setter,但是在內(nèi)部它們讓Vue 追蹤依賴(lài),在屬性被訪(fǎng)問(wèn)和修改時(shí)通知變化。
vue的數(shù)據(jù)雙向綁定
將MVVM作為數(shù)據(jù)綁定的入口,整合Observer,Compile和Watcher三者,通過(guò)Observer來(lái)監(jiān)聽(tīng)自己的model的數(shù)據(jù)變化,通過(guò)Compile來(lái)解析編譯模板指令(vue中是用來(lái)解析{{}}),最終利用watcher搭起observer和Compile之間的通信橋梁,達(dá)到數(shù)據(jù)變化—>視圖更新;視圖交互變化(input)—>數(shù)據(jù)model變更雙向綁定效果。
對(duì)vue-雙向數(shù)據(jù)的分析?/v-model 雙向數(shù)據(jù)綁定的原理
代碼如下(示例):
<script>
// Object 大小寫(xiě) value 書(shū)寫(xiě)
let data = {
name: '李白',
age: 18
}
Object.keys(data).forEach(key => {
defineReactiveProperty(data, key, data[key])
})
function defineReactiveProperty(data, key, value) {
Object.defineProperty(data, key, {
// get獲取
get() {
return value
},
// set 賦值
set(newVaue) {
if (newVaue === value) {
return
}
value = newVaue
compine()
}
})
}
compine()
</script>
</body>
</html>
function compine () {
// 通過(guò)document.querySelect('#app').childNodes 獲取app下所有的子元素
const nodes = document.querySelector('#app').childNodes
// 輸出一下這個(gè)值 當(dāng)前這個(gè)值是一個(gè)層級(jí)嵌套的數(shù)組我們通過(guò)foreach
// console.log(nodes)
nodes.forEach(item => {
// 再輸出一下item html:49 <input type="text" v-model="name"> 是一個(gè)input 輸入框
// console.log(item)
// 篩選出當(dāng)前是標(biāo)簽的 ,因?yàn)閚odes這個(gè)輸出會(huì)將空格以‘text' nodeType為3,而標(biāo)簽nodetype是1,if判斷篩選出是標(biāo)簽的
if (item.nodeType === 1){
const attrs = item.attributes
// console.log(attrs) {0: type, 1: v-model, type: type, v-model: v-model, length: 2} 返回了一個(gè)是數(shù)組
Array.from(attrs).forEach( arr => {
// console.log(arr) // texgt= 'text' v-mode: 'name' ,篩選出這個(gè)v-model
if (arr.nodeName === 'v-model'){
item.value = data[arr.nodeValue]
item.addEventListener('input',e => {
console.log(e.target.value)
//
data[arr.nodeValue] = e.target.value
})
}
})
}
})
}
總結(jié)

- 數(shù)據(jù)響應(yīng)式的實(shí)現(xiàn)無(wú)非是對(duì)象屬性攔截,我們使用 Object.defineProperty 來(lái)實(shí)現(xiàn),在vue3中使
用 Proxy 對(duì)象代理方案進(jìn)行了優(yōu)化 - 面試寶典上提到的幾個(gè)專(zhuān)業(yè)名詞
observe 對(duì)象指的是把數(shù)據(jù)處理成響應(yīng)式的對(duì)象
watcher 指的其實(shí)就是數(shù)據(jù)變化之后的更新函數(shù) (vue中的watcher有兩種,一種是用來(lái)更新視圖的watcher,一種是通過(guò)watch配置項(xiàng)聲明的watcher)
dep 指的就是使用發(fā)布訂閱實(shí)現(xiàn)的收集更新函數(shù)和觸發(fā)更新函數(shù)的對(duì)象 - 指令實(shí)現(xiàn)的核心無(wú)非是通過(guò)模板編譯找到標(biāo)識(shí)然后把數(shù)據(jù)綁上去,等到數(shù)據(jù)變化之后再重新放一次
- 發(fā)布訂閱模式的本質(zhì)是解決一對(duì)多的問(wèn)題,在vue中實(shí)現(xiàn)數(shù)據(jù)變化之后的精準(zhǔn)更新
到此這篇關(guān)于vue響應(yīng)式原理與雙向數(shù)據(jù)的文章就介紹到這了,更多相關(guān)vue響應(yīng)式原理與雙向數(shù)據(jù)內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Vue3父子組件互調(diào)方法的實(shí)現(xiàn)
本文主要介紹了Vue3父子組件互調(diào)方法的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2022-04-04
通過(guò)GASP讓vue實(shí)現(xiàn)動(dòng)態(tài)效果實(shí)例代碼詳解
GASP是一個(gè)JavaScript動(dòng)畫(huà)庫(kù),它支持快速開(kāi)發(fā)高性能的 Web 動(dòng)畫(huà)。GASP 使我們能夠輕松輕松快速的將動(dòng)畫(huà)串在一起,來(lái)創(chuàng)造一個(gè)高內(nèi)聚的流暢動(dòng)畫(huà)序列。這篇文章主要介紹了通過(guò)GASP讓vue實(shí)現(xiàn)動(dòng)態(tài)效果,需要的朋友可以參考下2019-11-11
Vue?+?ElementUI表格內(nèi)實(shí)現(xiàn)圖片點(diǎn)擊放大效果的兩種實(shí)現(xiàn)方式
這篇文章主要介紹了Vue?+?ElementUI表格內(nèi)實(shí)現(xiàn)圖片點(diǎn)擊放大效果的兩種實(shí)現(xiàn)方式,第一種使用el-popover彈出框來(lái)實(shí)現(xiàn)而第二種使用v-viewer插件實(shí)現(xiàn),需要的朋友可以參考下2024-08-08
Vue Element UI + OSS實(shí)現(xiàn)上傳文件功能
這篇文章主要為大家詳細(xì)介紹了Vue Element UI + OSS實(shí)現(xiàn)上傳文件功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-07-07
Vue3中關(guān)于ref和reactive的區(qū)別分析
這篇文章主要介紹了vue3關(guān)于ref和reactive的區(qū)別分析,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)吧2023-06-06
使用Vue 實(shí)現(xiàn)滑動(dòng)驗(yàn)證碼功能
本文章主要來(lái)介紹一下第一個(gè)階段,也就是前端校驗(yàn)的驗(yàn)證碼的實(shí)現(xiàn),下面來(lái)介紹一下拖動(dòng)驗(yàn)證碼的具體實(shí)現(xiàn)。這篇文章主要介紹了利用 Vue 實(shí)現(xiàn)滑動(dòng)驗(yàn)證碼,需要的朋友可以參考下2019-06-06

