Vue3響應(yīng)式對(duì)象是如何實(shí)現(xiàn)的(1)
簡(jiǎn)單的響應(yīng)式實(shí)現(xiàn)
為了方便說明,先來看一個(gè)簡(jiǎn)單的例子。
const obj = { text: 'hello vue' }
function effect() {
document.body.innerText = obj.text
}這段代碼中,如果obj是一個(gè)響應(yīng)式數(shù)據(jù),會(huì)產(chǎn)生什么效果呢?當(dāng)obj.text中的內(nèi)容改變時(shí),document.body.innerText也會(huì)隨之改變,從而修改頁(yè)面上顯示的內(nèi)容。因此,如果僅從這個(gè)簡(jiǎn)單的例子出發(fā),在修改obj后,再次執(zhí)行effect(),就能夠?qū)崿F(xiàn)響應(yīng)式。此時(shí),effect()被稱為副作用函數(shù),其副作用體現(xiàn)在document.body.innerText這個(gè)非effect()函數(shù)作用域內(nèi)的值被修改了。
實(shí)際使用中,響應(yīng)式實(shí)現(xiàn)要復(fù)雜的多。為了進(jìn)一步實(shí)現(xiàn)響應(yīng)式,讓我們基于上文的例子,再來捋一捋響應(yīng)式實(shí)現(xiàn)需要什么。我們需要在修改響應(yīng)式數(shù)據(jù)后觸發(fā)副作用函數(shù)。仔細(xì)思考這句話,這里其實(shí)要求了兩個(gè)功能:
- 當(dāng)
obj被讀取時(shí),收集副作用函數(shù); - 當(dāng)
obj被修改時(shí),觸發(fā)收集的副作用函數(shù);
問題更清晰了,我們只需要攔截讀取和修改操作分別進(jìn)行收集和觸發(fā),就能夠?qū)崿F(xiàn)響應(yīng)式了。這里我們可以使用Proxy
Proxy與響應(yīng)式
為什么需要Proxy?
Vue3的核心特征之一就是響應(yīng)式,而實(shí)現(xiàn)數(shù)據(jù)響應(yīng)式依賴于底層的Proxy。因此,想要完成Vue的響應(yīng)式功能,首先需要理解Proxy。
以reactive為例,當(dāng)想要?jiǎng)?chuàng)建一個(gè)響應(yīng)式對(duì)象時(shí),僅需要使用reactive對(duì)原始對(duì)象進(jìn)行包裹。
例如:
const user = reactive({
age: 25
})在Vue3的官方文檔中,對(duì)reactive的描述是:返回對(duì)象的響應(yīng)式副本。既然是響應(yīng)式副本,就需要產(chǎn)生一個(gè)與原始對(duì)象含有相同內(nèi)容的響應(yīng)式對(duì)象,之后使用這個(gè)響應(yīng)式對(duì)象替代原始對(duì)象,代替原始對(duì)象去做事情。這就體現(xiàn)了Proxy的價(jià)值——創(chuàng)建原始對(duì)象的代理。
Proxy創(chuàng)建的代理對(duì)象與原始對(duì)象有何不同?
先來看看如何使用Proxy創(chuàng)建代理對(duì)象。
let proxy = new Proxy(target, handler)
Proxy可以接收兩個(gè)參數(shù),其中,target表示被代理的原始對(duì)象,handler表示代理配置,代理配置中的捕捉器允許我們攔截并重新定義針對(duì)代理對(duì)象的操作。因此,如果在Proxy中,不進(jìn)行任何代理配置,Proxy僅會(huì)成為原始對(duì)象的一個(gè)透明包裝器(僅創(chuàng)建原始對(duì)象副本,但不產(chǎn)生任何其它功能)。因此,為了利用Proxy實(shí)現(xiàn)響應(yīng)式對(duì)象,我們必須使用Proxy中的代理配置。
在JavaScript規(guī)范中,存在一個(gè)描述JavaScript運(yùn)行機(jī)制的“內(nèi)部方法”。其中,讀取屬性的操作被稱為[[Get]],設(shè)置屬性的操作被稱為[[Set]]。通常,我們無(wú)法直接使用這些“內(nèi)部方法”,但Proxy給我們提供了捕捉器,使得對(duì)讀取和設(shè)置操作的攔截成為可能。
const p = new Proxy(obj, {
get(target, property, receiver) {/*攔截讀取屬性操作*/}
set(target, property, value, receiver) {/*攔截設(shè)置屬性操作*/}
})其中,target為被代理的原始對(duì)象,property為原始對(duì)象的屬性名,value為設(shè)置目標(biāo)屬性的值receiver是屬性所在的this對(duì)象,即Proxy代理對(duì)象。
Proxy是由JavaSciprt引擎實(shí)現(xiàn)的ES6特性,因此,沒有ES5的polyfill,也無(wú)法使用Babel轉(zhuǎn)譯。這也是Vue3不支持低版本IE的主要原因。
let fn
const data = { text: 'hello vue' }
const obj = new Proxy(data, {
get(target, key) {
fn = effect
return target[key]
},
set(target, key, newValue) {
target[key] = newValue
fn()
return true
}
})多副作用函數(shù)的響應(yīng)式實(shí)現(xiàn)
如果響應(yīng)式數(shù)據(jù)有多個(gè)相關(guān)的副作用函數(shù)應(yīng)該如何處理。
在上例中,存儲(chǔ)一個(gè)副作用函數(shù)我們可以使用一個(gè)fn進(jìn)行緩存,多個(gè)副作用函數(shù),我們?cè)谑占瘯r(shí)用數(shù)組把它裝起來就好,需要觸發(fā)時(shí),再遍歷數(shù)組挨個(gè)觸發(fā)收集到的副作用函數(shù)。
const fns = []
const data = { text: 'hello vue' }
const obj = new Proxy(data, {
get(target, key) {
fns.push(effect)
return target[key]
},
set(target, key, newValue) {
target[key] = newValue
fns.forEach(fn => fn())
return true
}
})這樣可以嗎?可以是可以,功能上已經(jīng)實(shí)現(xiàn)了。接下來讓我們來看看有沒有哪些地方可以優(yōu)化。
第一個(gè)可以優(yōu)化的點(diǎn)是,需要考慮被重復(fù)收集的函數(shù)。
舉個(gè)例子:
const effect1 = () => {
document.body.innerText = obj.text
}
const effect2 = effect1在這個(gè)例子中,effect1與effect2都表示同一個(gè)函數(shù),但在收集時(shí)會(huì)被重復(fù)收集,執(zhí)行時(shí)也會(huì)被重復(fù)執(zhí)行。這些重復(fù)顯然是不必要的,因此,我們需要在收集副作用函數(shù)的時(shí)候,干掉重復(fù)的函數(shù)。去重可以考慮Set。
const fns = new Set()
const data = { text: 'hello vue' }
const obj = new Proxy(data, {
get(target, key) {
fns.push(effect)
return target[key]
},
set(target, key, newValue) {
target[key] = newValue
fns.forEach(fn => fn())
return true
}
})不知道大家是否注意到,為了演示方便,我們?cè)谑占貢r(shí),都默認(rèn)被收集的副作用函數(shù)名是effect。但實(shí)際開發(fā)中,函數(shù)名肯定不會(huì)是固定的或是有規(guī)律可循的,我們肯定不能按函數(shù)名一個(gè)個(gè)手動(dòng)收集,因此,我們要想一種方式能夠不依賴于函數(shù)名進(jìn)行副作用函數(shù)收集。為了實(shí)現(xiàn)這一點(diǎn),我們將副作用函數(shù)進(jìn)行包裹,用一個(gè)activeEffect統(tǒng)一注冊(cè)。
let activeEffect
function effect(fn) {
activeEffect = fn
fn()
}
const fns = new Set()
const data = { text: 'hello vue' }
const obj = new Proxy(data, {
get(target, key) {
if(activeEffect) {
fns.add(activeEffect)
}
return target[key]
},
set(target, key, newValue) {
target[key] = newValue
fns.forEach(fn => fn())
return true
}
})
function fn() {
document.body.innerText = obj.text
}
effect(fn)到這里,我們已經(jīng)把容易想到的優(yōu)化都處理了。但這還沒完,這里還有個(gè)隱蔽的點(diǎn)沒有考慮到。此時(shí),我們的響應(yīng)式粒度還不夠細(xì),副作用函數(shù)的收集和觸發(fā)的最小單位是響應(yīng)式對(duì)象,這會(huì)導(dǎo)致不必要的副作用函數(shù)更新。例如,我們給出一個(gè)obj上不存在的屬性obj.notExist,對(duì)其進(jìn)行賦值操作。為了方便演示,我們?cè)?code>fn()中加一條打印語(yǔ)句,觀察結(jié)果。
function fn() {
document.body.innerText = obj.text
console.log('Done!')
}
可以看到,副作用函數(shù)被觸發(fā)了。這里obj.notExist屬性在obj上是不存在的,更談不上對(duì)應(yīng)了哪個(gè)副作用函數(shù),因此,這里的副作用函數(shù)不應(yīng)該被觸發(fā),或者換句話說,副作用函數(shù)僅能被對(duì)應(yīng)的響應(yīng)式對(duì)象的屬性影響,且僅在該屬性被修改時(shí)觸發(fā)。
更細(xì)粒度的觸發(fā)條件就要求我們收集副作用函數(shù)時(shí),針對(duì)響應(yīng)式對(duì)象的屬性進(jìn)行收集。聽上去有點(diǎn)無(wú)從下手,讓我們?cè)賮磙垡晦?。既然是響?yīng)式對(duì)象,首先還是要保證利用響應(yīng)式對(duì)象的Proxy來進(jìn)行收集和觸發(fā),但在收集的時(shí)候,就不能一股腦的把所有屬性對(duì)應(yīng)的副作用函數(shù)塞到一塊了,需要把誰(shuí)是誰(shuí)的分清楚(即一個(gè)屬性對(duì)應(yīng)了哪些副作用函數(shù)),讓每個(gè)屬性擁有自己收集和觸發(fā)副作用函數(shù)的能力,具體對(duì)應(yīng)關(guān)系如圖所示。

讓我們從后往前看。首先,屬性的副作用函數(shù)收集器我們可以沿用上文的思路,使用Set實(shí)現(xiàn)。再往前,每個(gè)屬性都需要配備一個(gè)副作用函數(shù)收集器,這是一個(gè)一對(duì)一的關(guān)系,我們可以使用將屬性值作為key,副作用收集器作為value,使用Map實(shí)現(xiàn),而這個(gè)Map就包含了單個(gè)響應(yīng)式對(duì)象的全部?jī)?nèi)容。
回看我們上面的代碼,其實(shí)我們?cè)谶M(jìn)行副作用函數(shù)收集的時(shí)候,僅使用了一個(gè)全局的容器承載。在實(shí)際開發(fā)中,我們需要盡量避免大量出現(xiàn)全局定義。因此,我們將響應(yīng)式對(duì)象放進(jìn)一個(gè)全局容器中統(tǒng)一管理。我們需要將每一個(gè)對(duì)象和其對(duì)應(yīng)的Map建立一對(duì)一的映射關(guān)系。整體關(guān)系如下圖所示:

按圖中思路,我們?cè)诖a中重新組織數(shù)據(jù)結(jié)構(gòu)。
let activeEffect
function effect(fn) {
activeEffect = fn
fn()
}
const objsMap = new Map() // 全局容器
const data = { text: 'hello vue' }
const obj = new Proxy(data, {
get(target, key) {
if(!activeEffect) return
let propsMap = objsMap.get(target) // 屬性容器
if(!propsMap) {
objsMap.set(target, (propsMap = new Map()))
}
let fns = propsMap.get(key) // 副作用函數(shù)收集器
if(!fns) {
propsMap.set(key, (fns = new Set()))
}
fns.add(activeEffect)
return target[key]
},
set(target, key, newValue) {
target[key] = newValue
const propsMap = objsMap.get(target) // 屬性容器
if(!propsMap) return
const fns = propsMap.get(key) // 副作用函數(shù)收集器
fns && fns.forEach(fn => fn())
return true
}
})
function fn() {
document.body.innerText = obj.text
console.log('Done!')
}
effect(fn)執(zhí)行我們更新的代碼,我們就能避免無(wú)效更新,只觸發(fā)與屬性相關(guān)的更新了。

到這結(jié)束了嗎?我們其實(shí)還能再做一點(diǎn)優(yōu)化。如果一個(gè)響應(yīng)式對(duì)象沒有被引用時(shí),說明這個(gè)對(duì)象不被使用,不被使用的對(duì)象應(yīng)該被JavaScript的垃圾回收器回收,避免造成內(nèi)存占用。但Map的特性(Map會(huì)被其key值持續(xù)引用)決定了,即使沒有任何引用,Map中的對(duì)象也不能被垃圾回收器回收。解決這個(gè)問題,只需要用WeakMap來代替Map,WeakMap的key是弱引用。到這里我們終于大功告成了,把功能實(shí)現(xiàn)完了。
功能實(shí)現(xiàn)完接下來要干嘛?對(duì),要重構(gòu),要看有沒有能重構(gòu)的地方。get和set中的內(nèi)容有點(diǎn)臃腫了,想想一開始說的,get時(shí)收集,set時(shí)觸發(fā)。最后,就讓我們來抽離封裝這兩部分。
let activeEffect
function effect(fn) {
activeEffect = fn
fn()
}
const objsMap = new WeakMap()
const data = { text: 'hello vue' }
const obj = new Proxy(data, {
get(target, key) {
track(target, key)
return target[key]
},
set(target, key, newValue) {
target[key] = newValue
trigger(target, key)
return true
}
})
// 收集
function track(target, key) {
if(!activeEffect) return
let propsMap = objsMap.get(target)
if(!propsMap) {
objsMap.set(target, (propsMap = new Map()))
}
let fns = propsMap.get(key)
if(!fns) {
propsMap.set(key, (fns = new Set()))
}
fns.add(activeEffect)
}
// 觸發(fā)
function trigger(target, key) {
const propsMap = objsMap.get(target)
if(!propsMap) return
const fns = propsMap.get(key)
fns && fns.forEach(fn => fn())
}
function fn() {
document.body.innerText = obj.text
console.log('Done!')
}
effect(fn)到此這篇關(guān)于Vue3響應(yīng)式對(duì)象是如何實(shí)現(xiàn)的的文章就介紹到這了,更多相關(guān)Vue3響應(yīng)式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
關(guān)于vuex強(qiáng)刷數(shù)據(jù)丟失問題解析
這篇文章主要介紹了關(guān)于vuex強(qiáng)刷數(shù)據(jù)丟失問題解析,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-04-04
vue2實(shí)現(xiàn)數(shù)據(jù)請(qǐng)求顯示loading圖
這篇文章主要為大家詳細(xì)介紹了vue2實(shí)現(xiàn)數(shù)據(jù)請(qǐng)求顯示loading圖,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11
vue實(shí)現(xiàn)圖片下載點(diǎn)擊按鈕彈出本地窗口選擇自定義保存路徑功能
vue前端實(shí)現(xiàn)前端下載,并實(shí)現(xiàn)點(diǎn)擊按鈕彈出本地窗口,選擇自定義保存路徑,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2023-12-12
Vue WatchEffect函數(shù)創(chuàng)建高級(jí)偵聽器
watchEffect傳入的函數(shù)會(huì)被立即執(zhí)行一次,并且在執(zhí)行的過程中會(huì)收集依賴;其次,只有收集的依賴發(fā)生變化時(shí),watchEffect傳入的函數(shù)才會(huì)再次執(zhí)行2023-03-03

