Vue3?Reactive響應(yīng)式原理邏輯詳解
前言
本篇文章主要講解vue響應(yīng)式原理的邏輯,也就是vue怎么從最開始一步步推導(dǎo)出響應(yīng)式的結(jié)構(gòu)框架。 先從頭構(gòu)建一個(gè)簡單函數(shù)推導(dǎo)出Vue3的Reactive原理,最后再進(jìn)行源碼的驗(yàn)證。
一、怎么實(shí)現(xiàn)變量變化
怎么實(shí)現(xiàn)變量變化,相關(guān)依賴的結(jié)果也跟著變化

當(dāng)原本price=5變?yōu)?code>price=20后total應(yīng)該變?yōu)?code>40,但是實(shí)際total并不會(huì)改變。 解決辦法可以這樣,當(dāng)變量改變了,重新計(jì)算一次,那么結(jié)果就會(huì)改變?yōu)樽钚碌慕Y(jié)果。
如果需要重新計(jì)算,我們需要將total語句存儲(chǔ)為一個(gè)函數(shù),才能實(shí)現(xiàn)依賴的變量改變就進(jìn)行一次依賴項(xiàng)計(jì)算。這里就用effect表示函數(shù)名。
來,試一下:

??,實(shí)現(xiàn)了變量price改變,依賴變量price quantity的變量total也發(fā)生改變。
下一步,我們要解決的問題是:應(yīng)該怎么把effect存儲(chǔ)起來,讓代碼更加有通用性,而不是一直復(fù)寫effect,分離出其他的功能的函數(shù)各司其職,也就是大家常說的解耦。
二、怎么實(shí)現(xiàn)變量變化
怎么實(shí)現(xiàn)變量變化,變量改變后就取出effect執(zhí)行
用什么存儲(chǔ)effect呢?當(dāng)然是用Set,因?yàn)镾et會(huì)過濾出重復(fù)的元素,所以能夠保證存儲(chǔ)在Set中的函數(shù)不是重復(fù)的。 這里定義一個(gè)存儲(chǔ)effect依賴的變量為dep = new Set(),定義track函數(shù)表示存儲(chǔ)的過程。 定義trigger函數(shù)用以取出dep中相關(guān)的effect函數(shù)執(zhí)行(這里定義的函數(shù)與Vue3源碼同名同意義)。
effect: 會(huì)影響結(jié)果的函數(shù)(要實(shí)現(xiàn)響應(yīng)式的依賴語句)track:保存所有的effecttrigger: 當(dāng)變量改變重新執(zhí)行代碼

??,解耦之后代碼結(jié)構(gòu)更清晰了。
下面需要解決的一個(gè)問題:一個(gè)object通常有多個(gè)屬性,比如product = { price: 5, quantity: 2 },在保存依賴時(shí)只創(chuàng)建了一個(gè)dep的集合,應(yīng)該給price和quantity都創(chuàng)建dep,因?yàn)?code>total的最終結(jié)果依賴這兩個(gè)屬性,其中任何一個(gè)改變都要觸發(fā)trigger函數(shù)。創(chuàng)建了兩個(gè)dep就需要一個(gè)容器將dep存儲(chǔ)起來。
三、將多個(gè)dep存儲(chǔ)在Map中
因?yàn)椴煌膶傩悦凶约簩?yīng)的dep,所以我們用Map結(jié)構(gòu)(鍵值對形式)來保存不同dep。

??,一個(gè)object的多個(gè)屬性依賴問題解決,更具有通用性了。
下一個(gè)問題是:不可能只有一個(gè)對象,多個(gè)對象又怎么辦?let product = { price: 5, quantity: 2 } let user = { firstName: "Joe", lastName: "Smith" },比如兩個(gè)對象的時(shí)候就需要進(jìn)一步修改上面的代碼了。
四、將多個(gè)object的depsMap繼續(xù)存儲(chǔ)起來
這里用WeakMap數(shù)據(jù)結(jié)構(gòu)去存儲(chǔ)多個(gè)需要響應(yīng)式的object的depsMap。WeakMap的基本使用和Map差不多,只不過WeakMap只接受對象為鍵值,而depsMap是一個(gè)Map結(jié)構(gòu)剛好(必須是)是對象類型。targetMap作為存儲(chǔ)多個(gè)depsMap的容器名。

??,到這里已經(jīng)基本實(shí)現(xiàn)了通用性的響應(yīng)式代碼了,但是還有最后一個(gè)問題就是:我們的代碼都需要手動(dòng)執(zhí)行(自己添加trigger運(yùn)行),不能自動(dòng)運(yùn)行。怎么讓它能夠自動(dòng)檢測變量改變,然后自動(dòng)修改結(jié)果呢?
五、核心
通過Reflect和Proxy解決自執(zhí)行問題
在JavaScript中,自動(dòng)檢測變量不就是get、自動(dòng)修改變量不就是set嗎?在Vue2.x版本中用ES5的Obeject.defineProperty()自帶的getter/setter去解決這個(gè)問題。ES6中Proxy也能解決這個(gè)問題,但是Proxy不兼任IE瀏覽器,當(dāng)時(shí)大家還討論過說不知道尤大怎么去考慮這個(gè)問題,現(xiàn)在問題的答案就是——不考慮。也就是根本不考慮IE兼不兼容????。
Proxy就是代理的意思,任何對真實(shí)數(shù)據(jù)的操作它都能攔截并且代理操作,也就是說Object上一些能實(shí)現(xiàn)的方法,Proxy也能實(shí)現(xiàn)。Proxy使用語法是new Proxy(target, hanler),handler是你想實(shí)現(xiàn)什么樣的代理功能配置。 而Reflect就更神奇了,它的作用是取代Object類上的一些方法讓Obeject類更純粹的代表一個(gè)類,不要附加太多方法在上面,比如a in obj表示判斷obj中是否有a,在Reflect中用Reflect.has(a)比較語義化的方式就可以代替之前的方法。
正是因?yàn)檫@樣,Proxy和Reflect就對應(yīng)上了,都有Object上的方法。
稍微封裝一下我們的函數(shù),名叫Reactive

??,至此,Vue3基本的響應(yīng)式原理就解析完了。
六、源碼解析(TypeScript)

return了createReactiveObject函數(shù),所以去看createReactiveObject。

前面的代碼都是判斷各種情況,我們就看最后幾行
const observed = new Proxy(
target,
collectionTypes.has(target.constructor) ? collectionHandlers : baseHandlers
)
可以看到Proxy的handler為collectionHandlers或者 baseHandlers,繼續(xù)選擇一個(gè)看一看。
在 baseHandlers中可以看到導(dǎo)出了get/set/deleteProperty等屬性配置:

我們看一下set:

和我們之前的邏輯差不多,只不過真正實(shí)現(xiàn)就很復(fù)雜,因?yàn)橛泻芏鄰?fù)雜條件需要去處理。
其他的get等方法也一樣,做了很多條件判斷處理,完善了每一種會(huì)出現(xiàn)的情況。
到此這篇關(guān)于Vue3 Reactive響應(yīng)式原理邏輯詳解的文章就介紹到這了,更多相關(guān)Vue3 Reactive響應(yīng)式 內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
vuejs 動(dòng)態(tài)添加input框的實(shí)例講解
今天小編就為大家分享一篇vuejs 動(dòng)態(tài)添加input框的實(shí)例講解,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2018-08-08
詳解用vue.js和laravel實(shí)現(xiàn)微信支付
本篇文章主要介紹了用vue.js和laravel實(shí)現(xiàn)微信支付,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-06-06
Vue中watch監(jiān)聽屬性新舊值相同的問題解決方案
這篇文章主要給大家分享了Vue中watch監(jiān)聽屬性新舊值相同問題解決方案,如果有遇到相同問題的朋友,可以參考閱讀本文2023-08-08
vue2手機(jī)APP項(xiàng)目添加開屏廣告或者閃屏廣告
這篇文章主要為大家詳細(xì)介紹了vue2手機(jī)APP項(xiàng)目添加開屏廣告或者閃屏廣告的方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-11-11
使用vue深度選擇器修改ElementUI組件內(nèi)樣式的示例代碼
在帶有scoped屬性的style中書寫樣式時(shí),無法作用影響到子組件中的樣式,此時(shí)我們會(huì)使用到deep深度選擇器,來解決此問題,我們在使用less預(yù)處理器,能正常使用,但是在scss預(yù)處理器中會(huì)報(bào)錯(cuò),下面通過本文介紹vue深度選擇器修改ElementUI組件內(nèi)樣式,需要的朋友可以參考下2022-12-12
vue3中nextTick()應(yīng)用實(shí)例詳解
這篇文章主要給大家介紹了關(guān)于vue3中nextTick()應(yīng)用的相關(guān)資料,nextTick()等待下一次DOM更新刷新的工具方法,文中通過代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-12-12
vue如何通過ref調(diào)用router-view子組件的方法
這篇文章主要介紹了vue?通過ref調(diào)用router-view子組件的方法,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友參考下吧2023-11-11
VUEJS實(shí)戰(zhàn)之利用laypage插件實(shí)現(xiàn)分頁(3)
這篇文章主要為大家詳細(xì)介紹了VUEJS實(shí)戰(zhàn)之修復(fù)錯(cuò)誤并且美化時(shí)間,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-06-06

