vue3響應式Proxy與Reflect的理解及基本使用實例詳解
正文
在第四章中,作者講述了Vue.js中響應式系統(tǒng)的設計與實現(xiàn),這一塊其實是整個框架的基石,也是MVVM中,ViewModel(VM)的重要組成部分。 其實在上一章中我已經感覺很難了,有一些操作作者也只是幾筆帶過,卻很值得我們思考。這一張中,我們將目光著重于響應式數(shù)據本身,來完善上一章中我們的demo。
理解Proxy與Reflect
vue3的響應式離不開Proxy,說到Proxy則離不開Reflect.這兩個對象是ES6新增的對象,同時在編程領域,他們也代表著2種設計模式,即代理與反射。設計模式都是后話,等我學會了也一定寫一篇文章分享一下??.
Proxy
Proxy 可以理解成,在目標對象之前架設一層“攔截”,外界對該對象的訪問,都必須經過這層攔截,而我們就可以通過這層攔截去改變目標對象的內容或者行為,或者叫過濾和控制。這個詞本意就是代理,好比一個代理人站在神奇,我們所有行為都會被他過濾,可能我們說的話,經過代理人一說,意思就變了。
ES6 原生提供 Proxy 構造函數(shù),用來生成 Proxy 實例。
var proxy = new Proxy(target, handler);
其中target表示要代理的那個對象,handler則是表示我們需要攔截的行為,這里直接放一張阮一峰的截圖。

Reflect
Reflect中文譯為:反射。如果說Proxy 是有一個代理人站在身前面,幫你攔截并處理一些行為,那么Reflect就是你身后的一面鏡子,它能看見真實的自己。
而你自己,就是一個類或者對象,或者一個函數(shù),只要是js中存在的,都能被Proxy 和 Reflect處理。

它的操作和Proxy正好相反,但卻一一對應。比如我們獲取對象中一個屬性。
const obj = {foo:1}
const a = Reflect.get(obj, 'foo')
這一小節(jié)主要是介紹了Proxy與Reflect,后面會有一個應用老告訴你為什么Proxy與Reflect與響應式數(shù)據息息相關。
實踐示例
看完了Proxy與Reflect的基本使用之后,我們實踐一下。
我們曾經寫過這樣的代碼
const reactive = (object)=>{
return new Proxy(object,{
get(target,key){
track(target,key)
return target[key]
}
set(target,key, newVal){
target[key] = newVal
trigger(target,key)
return true
}
})
}
其實就是用Proxy代理了對象讀和取操作,在讀的時候收集依賴,在取的時候觸發(fā)響應??雌饋硭坪鯖]有問題,那么我們再試繼續(xù)往下寫
const obj = {
a:1,
get b(){
return this.a
}
}
const data = reactive(obj)
effect(()=>{
console.log(data.b)
})
setTimeOut(()=>{
data.b++
},500)
這里我們沒有用一般的對象寫法,而是通過訪問器為它新增了一個b屬性.之后,我們先把這個對象轉換為響應式對象,再給他們設定一個響應式的回調,然后在冬天改變他的值,理論上這時候應該會執(zhí)行副作用函數(shù),但是實際上呢,根本不會執(zhí)行。
我們回顧一下之前寫的reactive方法,在里面返回的是target[key],當我們的target是obj,key是b的時候,那個this會是誰呢?因為target是原始對象,也就是obj,根據誰調用是誰的原則,這個this也就指向了obj。obj是響應式對象嗎,顯然不是,那個b也就永遠不會執(zhí)行副作用函數(shù),響應式就失效了。
這里其實就是this的指向問題,你可能會說一般人怎么會用getter去賦值屬性呢,但是這個作為一個簡單的case,甚至都算不上邊界,我們需要解決它。
解決的方法也很簡單,就是通過Reflect。這也是為什么我說Proxy與Reflect就是焦不離孟 孟不離焦. 我們的reactive,get的時候,加入第三個參數(shù)receiver
get(target,key){
track(target,key,receiver)
return Reflect.get(target,key,receiver)
}
我這里理解的是,receiver就相當于函數(shù)的bind方法,它改變的this的執(zhí)行,當我們同過Reflect讀取值的時候,this的指向被改為receiver,而Reflect時的receiver又是Proxy中的入參,它執(zhí)行了這個Proxy,從而把前文中this的指向由obj改為data,這樣響應式就不會丟失了。
下一章中,作者會深入的介紹一下Proxy和一些相關的標準。
以上就是vue3響應式Proxy與Reflect的理解及基本使用實例詳解的詳細內容,更多關于vue3響應式Proxy Reflect的資料請關注腳本之家其它相關文章!
相關文章
vue3 el-form-item如何自定義label標簽內容
這篇文章主要介紹了vue3 el-form-item如何自定義label標簽內容問題,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-10-10
vscode使用Eslint+Prettier格式化代碼的詳細操作
這篇文章主要介紹了vscode使用Eslint+Prettier格式化代碼,本文通過圖文示例代碼相結合給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2022-08-08
使用自動導入后eslint報錯eslint9的問題及解決方法
本文介紹了使用`pnpm create vue@latest`創(chuàng)建Vue應用時,如何配置ESLint和Prettier,包括解決兩者沖突以及自動導入后Eslint報錯的問題,感興趣的朋友一起看看吧2025-03-03
vue3 onMounted異步函數(shù)同步請求async/await實現(xiàn)
這篇文章主要為大家介紹了vue3 onMounted初始化數(shù)據異步函數(shù)/同步請求async/await實現(xiàn)詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2023-07-07
vue中的addEventListener和removeEventListener用法說明
這篇文章主要介紹了vue中的addEventListener和removeEventListener用法說明,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2022-06-06

