溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

JS監(jiān)聽(tīng)變量改變?nèi)绾螌?shí)現(xiàn)

發(fā)布時(shí)間:2023-04-15 11:12:01 來(lái)源:億速云 閱讀:180 作者:iii 欄目:開(kāi)發(fā)技術(shù)

這篇文章主要介紹“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

    Object.defineProperty() 方法會(huì)直接在一個(gè)對(duì)象上定義一個(gè)新屬性,或者修改一個(gè)對(duì)象的現(xiàn)有屬性,并返回此對(duì)象

    ——MDN

    用法如下:

    Object.defineProperty(obj, prop, option)

    入?yún)⒂梅ǎ?/h4>
    • obj:代理對(duì)象;

    • prop:代理對(duì)象中的key;

    • option:配置對(duì)象,getset都在其中配置;

    例子:

    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

    對(duì)一個(gè)對(duì)象進(jìn)行整體響應(yīng)式監(jiān)聽(tīng):

    // 監(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

    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 ,一一嘗試一下,這里不再贅述

    在React中的實(shí)踐

    這里展示兩段偽代碼,大概業(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的彈窗。

    Object.defineProperty

    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

    Proxy

    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í)用的文章!

    向AI問(wèn)一下細(xì)節(jié)

    免責(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)容。

    js
    AI