您好,登錄后才能下訂單哦!
這篇文章主要介紹“JS監(jiān)聽(tīng)變量改變?nèi)绾螌?shí)現(xiàn)”,在日常操作中,相信很多人在JS監(jiān)聽(tīng)變量改變?nèi)绾螌?shí)現(xiàn)問(wèn)題上存在疑惑,小編查閱了各式資料,整理出簡(jiǎn)單好用的操作方法,希望對(duì)大家解答”JS監(jiān)聽(tīng)變量改變?nèi)绾螌?shí)現(xiàn)”的疑惑有所幫助!接下來(lái),請(qǐng)跟著小編一起來(lái)學(xué)習(xí)吧!
在業(yè)務(wù)中,由于項(xiàng)目采用微前端架構(gòu),需要通過(guò)A應(yīng)用的某個(gè)值的變化對(duì)B應(yīng)用中的DOM進(jìn)行改變(如彈出一個(gè)Modal),第一個(gè)想到的可能是發(fā)布訂閱模式,其實(shí)不如將問(wèn)題縮小化,采用原生的能力去解決。
下面給出兩種解決方案,同時(shí)也是尤大寫(xiě)Vue
時(shí)的思路
ES5 的 Object.defineProperty
ES6 的 Proxy
Object.defineProperty()
方法會(huì)直接在一個(gè)對(duì)象上定義一個(gè)新屬性,或者修改一個(gè)對(duì)象的現(xiàn)有屬性,并返回此對(duì)象——MDN
Object.defineProperty(obj, prop, option)
obj:代理對(duì)象;
prop:代理對(duì)象中的key;
option:配置對(duì)象,get
、set
都在其中配置;
var obj = { name: 'sorryhc' } var rocordName = 'sorryhc'; Object.defineProperty(obj, 'name', { enumerable: true, configurable:true, set: function(newVal) { rocordName = newVal console.log('set: ' + rocordName) }, get: function() { console.log('get: ' + rocordName) return rocordName } }) obj.name = 'sorrycc' // set: sorrycc console.log(obj.name) // get: sorrycc
// 監(jiān)視對(duì)象 function observe(obj) { // 遍歷對(duì)象,使用 get/set 重新定義對(duì)象的每個(gè)屬性值 Object.keys(obj).forEach(key => { defineReactive(obj, key, obj[key]) }) } function defineReactive(obj, k, v) { // 遞歸子屬性 if (typeof(v) === 'object') observe(v) // 重定義 get/set Object.defineProperty(obj, k, { enumerable: true, configurable: true, get: function reactiveGetter() { console.log('get: ' + v) return v }, // 重新設(shè)置值時(shí),觸發(fā)收集器的通知機(jī)制 set: function reactiveSetter(newV) { console.log('set: ' + newV) v = newV }, }) } let data = {a: 1} // 監(jiān)視對(duì)象 observe(data) data.a // get: 1 data.a = 2 // set: 2
整體思路就是遇到子對(duì)象就遞歸,和深拷貝一樣的讀參順序。
如果學(xué)習(xí)過(guò)Vue2
源碼的同學(xué)可能比較熟,基于下面的缺陷,也是出現(xiàn)了$set
、$get
的用法。
IE8 及更低版本 IE 是不支持的
無(wú)法檢測(cè)到對(duì)象屬性的新增或刪除
如果修改數(shù)組的 length
( Object.defineProperty
不能監(jiān)聽(tīng)數(shù)組的長(zhǎng)度),以及數(shù)組的 push
等變異方法是無(wú)法觸發(fā) setter
的
Proxy 對(duì)象用于創(chuàng)建一個(gè)對(duì)象的代理,從而實(shí)現(xiàn)基本操作的攔截和自定義(如屬性查找、賦值、枚舉、函數(shù)調(diào)用等)
— MDN
const obj = new Proxy(target, handler)
其中:
target
:要使用 Proxy
包裝的目標(biāo)對(duì)象(可以是任何類型的對(duì)象,包括原生數(shù)組,函數(shù),甚至另一個(gè)代理)
handler
:一個(gè)通常以函數(shù)作為屬性的對(duì)象,各屬性中的函數(shù)分別定義了在執(zhí)行各種操作時(shí)代理 obj
的行為
const handler = { get: function(target, name){ return name in target ? target[name] : 'no prop!' }, set: function(target, prop, value, receiver) { target[prop] = value; console.log('property set: ' + prop + ' = ' + value); return true; } }; var user = new Proxy({}, handler) user.name = 'sorryhc' // property set: name = sorryhc console.log(user.name) // sorryhc console.log(user.age) // no prop!
并且Proxy
提供了更豐富的代理能力:
getPrototypeOf
/ setPrototypeOf
isExtensible
/ preventExtensions
ownKeys
/ getOwnPropertyDescriptor
defineProperty
/ deleteProperty
get
/ set
/ has
apply
/ construct
感興趣的可以查看 MDN ,一一嘗試一下,這里不再贅述
這里展示兩段偽代碼,大概業(yè)務(wù)流程是,當(dāng)點(diǎn)擊頁(yè)面某個(gè)按鈕(打開(kāi)/關(guān)閉彈窗),觸發(fā)window.obj.showModal
的切換,從而被監(jiān)聽(tīng)到全局變量的變化,從而改變React
中的state
狀態(tài),最終觸發(fā)Modal
的彈窗。
window.obj = { showModal: false } const [visible, setVisible] = useState(false); useEffect(() => { visible && Modal.show({ // ... }) }, [visible]) Object.defineProperty(window.obj, 'showModal', { enumerable: true, configurable:true, set: function(newVal) { setVisible(newVal); console.log('set: ' + newVal) }, get: function() { console.log('get: ' + visible) return visible } }) window.obj.showModal = !window.obj.showModal // set: true console.log(window.obj.showModal) // get: true
const [visible, setVisible] = useState(false); useEffect(() => { visible && Modal.show({ // ... }) }, [visible]) const handler = { get: function(target, name){ return name in target ? target[name] : 'no prop!' }, set: function(target, prop, value, receiver) { target[prop] = value; setVisible(value); console.log('property set: ' + prop + ' = ' + value); return true; } }; window.obj = new Proxy({showModal: false}, handler) window.obj.showModal = !window.obj.showModal // property set: showModal = true console.log(window.obj.showModal) // true
到此,關(guān)于“JS監(jiān)聽(tīng)變量改變?nèi)绾螌?shí)現(xiàn)”的學(xué)習(xí)就結(jié)束了,希望能夠解決大家的疑惑。理論與實(shí)踐的搭配能更好的幫助大家學(xué)習(xí),快去試試吧!若想繼續(xù)學(xué)習(xí)更多相關(guān)知識(shí),請(qǐng)繼續(xù)關(guān)注億速云網(wǎng)站,小編會(huì)繼續(xù)努力為大家?guī)?lái)更多實(shí)用的文章!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。