JavaScript使用Proxy編寫(xiě)一個(gè)取值限制器
前言
最近一直在開(kāi)發(fā)低代碼平臺(tái)的東西,由于項(xiàng)目里面東西有點(diǎn)多,取值或調(diào)用起來(lái)比較麻煩,特別是那些實(shí)例的特殊函數(shù)和變量。
所以我實(shí)現(xiàn)了一個(gè)實(shí)例收集器,讓開(kāi)發(fā)者的權(quán)限變得非常高,細(xì)節(jié)就不聊了。
盡管這樣子會(huì)讓開(kāi)發(fā)的體驗(yàn)變得很簡(jiǎn)單,但是大家都知道最便宜的東西往往是最昂貴的,當(dāng)你可以隨意在項(xiàng)目中讀取到不同組件的實(shí)例的時(shí)候,很容易導(dǎo)致某一個(gè)組件和另一個(gè)風(fēng)馬牛不相及的組件產(chǎn)生關(guān)聯(lián),當(dāng)某一側(cè)不當(dāng)操作修了實(shí)例的某些內(nèi)容的話,很可能導(dǎo)致隱藏且嚴(yán)重的bug。
因此對(duì)于這種上帝的操作 必須進(jìn)行限制,因此對(duì)此我創(chuàng)建了一個(gè)取值限制器來(lái)進(jìn)行約束,代碼很簡(jiǎn)單,不妨一看。
源碼實(shí)現(xiàn)
// 初始化 限制proxy生成函數(shù) 的 構(gòu)造函數(shù) // fn 校驗(yàn)當(dāng)前對(duì)象是否可以讀取 const LimitProxyConstructorFn = (fn) => { // 限制proxy // target 需要劫持的對(duì)象 const createLimitProxy = (target) => { return new Proxy(target, { get(target, property) { const res = Reflect.get(target, property); const isLegal = fn(property); // 判斷當(dāng)前取值是否是對(duì)象,因?yàn)閷?duì)象可能要限制取值 if (typeof res === "object" && res !== null) { // 合法對(duì)象 返回一個(gè)禁止賦值但支持取值的proxy對(duì)象 if (isLegal) { return new Proxy(target, { get(target, property) { return Reflect.get(target, property); }, set() { throw new Error("無(wú)法寫(xiě)入屬性"); }, }); } // 如果不合法,則遞歸創(chuàng)建 限制proxy return createLimitProxy(res); } else { // 如果當(dāng)前項(xiàng)不是對(duì)象,判斷是否合法,合法則返回,反之異常 if (isLegal) return res; throw new Error("取值不合法"); } }, set() { throw new Error("無(wú)法寫(xiě)入屬性"); }, }); }; return createLimitProxy; };
測(cè)試用例
初始化限制proxy對(duì)象構(gòu)造函數(shù)
效果如下:
limitProxy 是一個(gè)proxy代理對(duì)象
看到藍(lán)色框里面的是取值操作,只有以__
開(kāi)頭的屬性才能讀取。
如果__
開(kāi)頭的屬性是一個(gè)對(duì)象則,該對(duì)象下所有的屬性都可以讀取。
反之認(rèn)為讀取不合法。
可以看到紅框里面都是賦值操作,限制proxy禁用了所有的賦值操作,保證原數(shù)據(jù)的安全
代碼解釋
這段代碼的作用是創(chuàng)建一個(gè)可以限制對(duì)象屬性讀取和寫(xiě)入操作的代理對(duì)象生成函數(shù)。傳入的驗(yàn)證函數(shù) fn
用于確定哪些屬性是合法的。
生成的代理對(duì)象會(huì)根據(jù)驗(yàn)證函數(shù)的結(jié)果,限制對(duì)屬性的讀取和寫(xiě)入操作。
代碼解釋?zhuān)?/p>
- 代碼定義了一個(gè)限制proxy函數(shù)的生成函數(shù)
LimitProxyConstructorFn
,該函數(shù)接受一個(gè)參數(shù)fn
,用于驗(yàn)證當(dāng)前對(duì)象是否可以讀取,fn
接受一個(gè)一個(gè)參數(shù)property
,即當(dāng)前讀取的對(duì)象的key的名稱(chēng),需要返回boolean
判斷是否合法。 - 函數(shù)內(nèi)部定義了一個(gè)名為
createLimitProxy
的函數(shù),它接受一個(gè)參數(shù)target
,表示需要劫持的對(duì)象。createLimitProxy
函數(shù)使用new Proxy()
創(chuàng)建一個(gè)代理對(duì)象來(lái)限制目標(biāo)對(duì)象的操作。
代理對(duì)象的 get
處理邏輯如下:
使用 Reflect.get()
獲取原始對(duì)象 target
的屬性值,并將結(jié)果存儲(chǔ)在變量 res
中。
調(diào)用驗(yàn)證函數(shù) fn
并傳入屬性名稱(chēng) property
,將結(jié)果存儲(chǔ)在變量 isLegal
中,用于判斷屬性是否合法。
如果取值 res
是對(duì)象(不包括 null
),則進(jìn)行進(jìn)一步處理:
- 如果對(duì)象合法(
isLegal
為true
),則返回一個(gè)新的代理對(duì)象,該對(duì)象禁止賦值但允許取值。這個(gè)新代理對(duì)象的get
處理邏輯與原始代理對(duì)象相同,即返回原始對(duì)象的屬性值,而set
處理邏輯拋出一個(gè)錯(cuò)誤,表示無(wú)法寫(xiě)入屬性。 - 如果對(duì)象不合法,則遞歸調(diào)用
createLimitProxy
函數(shù),創(chuàng)建針對(duì)該對(duì)象的限制代理,并返回限制代理對(duì)象。
如果取值 res
不是對(duì)象,則判斷屬性是否合法:
- 如果屬性合法,則直接返回取值
res
。 - 如果屬性不合法,則拋出一個(gè)錯(cuò)誤,表示取值不合法。
代理對(duì)象的 set
處理邏輯與上述相同,無(wú)論是對(duì)原始對(duì)象還是對(duì)代理對(duì)象進(jìn)行屬性賦值操作,都會(huì)拋出一個(gè)錯(cuò)誤,表示無(wú)法寫(xiě)入屬性。
構(gòu)造函數(shù) LimitProxyConstructorFn
返回內(nèi)部定義的 createLimitProxy
函數(shù),以便可以使用該函數(shù)來(lái)生成限制代理對(duì)象。
總結(jié)
總的來(lái)說(shuō)這個(gè)東西還挺有意思的,特別是那種需要限制取值權(quán)限的場(chǎng)景,代碼也很少,感興趣的朋友不妨了解一下,當(dāng)然代碼還有不少優(yōu)化空間。
到此這篇關(guān)于JavaScript使用Proxy編寫(xiě)一個(gè)取值限制器的文章就介紹到這了,更多相關(guān)JavaScript Proxy取值限制器內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
解決function函數(shù)內(nèi)的循環(huán)變量
鼠標(biāo)放到指定的行上自動(dòng)彈出當(dāng)前的個(gè)數(shù),從0開(kāi)始,這個(gè)功能方便我們?cè)趖ab切換中定位2008-10-10微信小程序如何根據(jù)不同用戶(hù)切換不同TabBar(簡(jiǎn)單易懂!)
小程序中我們可能需要根據(jù)不同的權(quán)限展示不同的tabbar,下面這篇文章主要給大家介紹了關(guān)于微信小程序如何根據(jù)不同用戶(hù)切換不同TabBar的相關(guān)資料,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-04-04element-plus 官方表格排序問(wèn)題小結(jié)
在使用Element Plus官方API時(shí),表格默認(rèn)排序可能會(huì)遇到問(wèn)題,一個(gè)列表可能被多次排序影響數(shù)據(jù)展示,解決方法是修改useSortTable.js文件,這樣可以確保表格按預(yù)期正確排序,更多詳情可查閱相關(guān)的技術(shù)文檔或資源2024-10-10js通過(guò)audioContext實(shí)現(xiàn)3D音效
這篇文章主要為大家詳細(xì)介紹了js通過(guò)audioContext實(shí)現(xiàn)3D音效,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2021-04-04JavaScript數(shù)組特性與實(shí)踐應(yīng)用深入詳解
這篇文章主要介紹了JavaScript數(shù)組特性與實(shí)踐應(yīng)用,較為深入而詳細(xì)的分析了javascript數(shù)組的功能、屬性、使用方法及操作注意事項(xiàng),需要的朋友可以參考下2018-12-12關(guān)于數(shù)據(jù)與后端進(jìn)行交流匹配(點(diǎn)亮星星)
這篇文章主要介紹了關(guān)于數(shù)據(jù)與后端進(jìn)行交流匹配(點(diǎn)亮星星) 的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-08-08JavaScript實(shí)現(xiàn)頁(yè)面無(wú)操作倒計(jì)時(shí)退出
這篇文章主要為大家詳細(xì)介紹了JavaScript實(shí)現(xiàn)頁(yè)面無(wú)操作倒計(jì)時(shí)退出,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-10-10Three.js中如何使用CSS3DRenderer和CSS3DSprite實(shí)現(xiàn)模型標(biāo)簽文字
在Three.js中,使用CSS3DRenderer和CSS3DSprite可以輕松地實(shí)現(xiàn)模型標(biāo)簽文字的效果,為場(chǎng)景中的模型提供更直觀的信息展示,本文將介紹如何使用這兩個(gè)工具來(lái)實(shí)現(xiàn)模型標(biāo)簽文字,并提供相應(yīng)的代碼示例,感興趣的朋友跟隨小編一起看看吧2024-05-05