国产精品电影_久久视频免费_欧美日韩国产激情_成年人视频免费在线播放_日本久久亚洲电影_久久都是精品_66av99_九色精品美女在线_蜜臀a∨国产成人精品_冲田杏梨av在线_欧美精品在线一区二区三区_麻豆mv在线看

監聽一個變量的變化,需要怎么做

安全 應用安全
監聽一個變量的變化,當變量變化時執行某些操作,這類似現在流行的前端框架(例如 React、Vue等)中的數據綁定功能,在數據更新時自動更新 DOM 渲染,那么如何實現數據綁定喃?

[[395785]]

本文轉載自微信公眾號「三分鐘學前端」,作者sisterAn。轉載本文請聯系三分鐘學前端公眾號。

監聽一個變量的變化,當變量變化時執行某些操作,這類似現在流行的前端框架(例如 React、Vue等)中的數據綁定功能,在數據更新時自動更新 DOM 渲染,那么如何實現數據綁定喃?

本文給出兩種思路:

  • ES5 的 Object.defineProperty
  • ES6 的 Proxy

ES5 的 Object.defineProperty

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

——MDN

  1. Object.defineProperty(obj, prop, descriptor) 

其中:

  • obj :要定義屬性的對象
  • prop :要定義或修改的屬性的名稱或 Symbol
  • descriptor :要定義或修改的屬性描述符
  1. var user = {  
  2.     name'sisterAn'  
  3.  
  4. Object.defineProperty(user'name', { 
  5.     enumerable: true
  6.     configurable:true
  7.     setfunction(newVal) { 
  8.         this._name = newVal  
  9.         console.log('set: ' + this._name) 
  10.     }, 
  11.     get: function() { 
  12.         console.log('get: ' + this._name) 
  13.         return this._name 
  14.     } 
  15. }) 
  16.  
  17. user.name = 'an' // set: an 
  18. console.log(user.name) // get: an 

如果是完整的對變量的每一個子屬性進行監聽:

  1. // 監視對象 
  2. function observe(obj) { 
  3.    // 遍歷對象,使用 get/set 重新定義對象的每個屬性值 
  4.     Object.keys(obj).map(key => { 
  5.         defineReactive(obj, key, obj[key]) 
  6.     }) 
  7.  
  8. function defineReactive(obj, k, v) { 
  9.     // 遞歸子屬性 
  10.     if (typeof(v) === 'object') observe(v) 
  11.      
  12.     // 重定義 get/set 
  13.     Object.defineProperty(obj, k, { 
  14.         enumerable: true
  15.         configurable: true
  16.         get: function reactiveGetter() { 
  17.             console.log('get: ' + v) 
  18.             return v 
  19.         }, 
  20.         // 重新設置值時,觸發收集器的通知機制 
  21.         setfunction reactiveSetter(newV) { 
  22.             console.log('set: ' + newV) 
  23.             v = newV 
  24.         }, 
  25.     }) 
  26.  
  27. let data = {a: 1} 
  28. // 監視對象 
  29. observe(data) 
  30. data.a // get: 1 
  31. data.a = 2 // set: 2 

通過 map 遍歷,通過深度遞歸監聽子子屬性

注意, Object.defineProperty 擁有以下缺陷:

  • IE8 及更低版本 IE 是不支持的
  • 無法檢測到對象屬性的新增或刪除
  • 如果修改數組的 length ( Object.defineProperty 不能監聽數組的長度),以及數組的 push 等變異方法是無法觸發 setter 的

對此,我們看一下 vue2.x 是如何解決這塊的?

vue2.x 中如何監測數組變化

使用了函數劫持的方式,重寫了數組的方法,Vue 將 data 中的數組進行了原型鏈重寫,指向了自己定義的數組原型方法。這樣當調用數組 api 時,可以通知依賴更新。如果數組中包含著引用類型,會對數組中的引用類型再次遞歸遍歷進行監控。這樣就實現了監測數組變化。

對于數組而言,Vue 內部重寫了以下函數實現派發更新

  1. // 獲得數組原型 
  2. const arrayProto = Array.prototype 
  3. export const arrayMethods = Object.create(arrayProto) 
  4. // 重寫以下函數 
  5. const methodsToPatch = [ 
  6.   'push'
  7.   'pop'
  8.   'shift'
  9.   'unshift'
  10.   'splice'
  11.   'sort'
  12.   'reverse' 
  13. methodsToPatch.forEach(function (method) { 
  14.   // 緩存原生函數 
  15.   const original = arrayProto[method] 
  16.   // 重寫函數 
  17.   def(arrayMethods, method, function mutator (...args) { 
  18.   // 先調用原生函數獲得結果 
  19.     const result = original.apply(this, args) 
  20.     const ob = this.__ob__ 
  21.     let inserted 
  22.     // 調用以下幾個函數時,監聽新數據 
  23.     switch (method) { 
  24.       case 'push'
  25.       case 'unshift'
  26.         inserted = args 
  27.         break 
  28.       case 'splice'
  29.         inserted = args.slice(2) 
  30.         break 
  31.     } 
  32.     if (inserted) ob.observeArray(inserted) 
  33.     // 手動派發更新 
  34.     ob.dep.notify() 
  35.     return result 
  36.   }) 
  37. }) 

vue2.x 怎么解決給對象新增屬性不會觸發組件重新渲染的問題

受現代 JavaScript 的限制 ( Object.observe 已被廢棄),Vue 無法檢測到對象屬性的添加或刪除。

由于 Vue 會在初始化實例時對屬性執行 getter/setter 轉化,所以屬性必須在 data 對象上存在才能讓 Vue 將它轉換為響應式的。

對于已經創建的實例,Vue 不允許動態添加根級別的響應式屬性。但是,可以使用 Vue.set(object, propertyName, value) 方法向嵌套對象添加響應式屬性。

vm.$set()實現原理

  1. export function set(target: Array<any> | Object, keyany, val: any): any { 
  2.   // target 為數組 
  3.   if (Array.isArray(target) && isValidArrayIndex(key)) { 
  4.     // 修改數組的長度, 避免索引>數組長度導致 splice() 執行有誤 
  5.     target.length = Math.max(target.length, key); 
  6.     // 利用數組的 splice 方法觸發響應式 
  7.     target.splice(key, 1, val); 
  8.     return val; 
  9.   } 
  10.   // target 為對象, key 在 target 或者 target.prototype 上 且必須不能在 Object.prototype 上,直接賦值 
  11.   if (key in target && !(key in Object.prototype)) { 
  12.     target[key] = val; 
  13.     return val; 
  14.   } 
  15.   // 以上都不成立, 即開始給 target 創建一個全新的屬性 
  16.   // 獲取 Observer 實例 
  17.   const ob = (target: any).__ob__; 
  18.   // target 本身就不是響應式數據, 直接賦值 
  19.   if (!ob) { 
  20.     target[key] = val; 
  21.     return val; 
  22.   } 
  23.   // 進行響應式處理 
  24.   defineReactive(ob.value, key, val); 
  25.   ob.dep.notify(); 
  26.   return val; 
  • 如果目標是數組,使用 vue 實現的變異方法 splice 實現響應式
  • 如果目標是對象,判斷屬性存在,即為響應式,直接賦值
  • 如果 target 本身就不是響應式,直接賦值
  • 如果屬性不是響應式,則調用 defineReactive 方法進行響應式處理

ES6 的 Proxy

眾所周知,尤大大的 vue3.0 版本用 Proxy 代替了defineProperty 來實現數據綁定,因為 Proxy 可以直接監聽對象和數組的變化,并且有多達 13 種攔截方法。并且作為新標準將受到瀏覽器廠商重點持續的性能優化。

Proxy

Proxy 對象用于創建一個對象的代理,從而實現基本操作的攔截和自定義(如屬性查找、賦值、枚舉、函數調用等)

— MDN

  1. const p = new Proxy(target, handler) 

其中:

  • target :要使用 Proxy 包裝的目標對象(可以是任何類型的對象,包括原生數組,函數,甚至另一個代理)
  • handler :一個通常以函數作為屬性的對象,各屬性中的函數分別定義了在執行各種操作時代理 p 的行為
  1. var handler = { 
  2.     get: function(target, name){ 
  3.         return name in target ? target[name] : 'no prop!' 
  4.     }, 
  5.     setfunction(target, prop, value, receiver) { 
  6.         target[prop] = value; 
  7.         console.log('property set: ' + prop + ' = ' + value); 
  8.         return true
  9.     } 
  10. }; 
  11.  
  12. var user = new Proxy({}, handler) 
  13. user.name = 'an' // property setname = an 
  14.  
  15. console.log(user.name) // an 
  16. console.log(user.age) // no prop! 

上面提到過 Proxy 總共提供了 13 種攔截行為,分別是:

  • getPrototypeOf / setPrototypeOf
  • isExtensible / preventExtensions
  • ownKeys / getOwnPropertyDescriptor
  • defineProperty / deleteProperty
  • get / set / has
  • apply / construct

感興趣的可以查看 MDN ,一一嘗試一下,這里不再贅述

另外考慮兩個問題:

  • Proxy只會代理對象的第一層,那么又是怎樣處理這個問題的呢?
  • 監測數組的時候可能觸發多次get/set,那么如何防止觸發多次呢(因為獲取push和修改length的時候也會觸發)

Vue3 Proxy

對于第一個問題,我們可以判斷當前 Reflect.get 的返回值是否為 Object ,如果是則再通過 reactive 方法做代理, 這樣就實現了深度觀測。

對于第二個問題,我們可以判斷是否是 hasOwProperty

下面我們自己寫個案例,通過proxy 自定義獲取、增加、刪除等行為

  1. const toProxy = new WeakMap(); // 存放被代理過的對象 
  2. const toRaw = new WeakMap(); // 存放已經代理過的對象 
  3. function reactive(target) { 
  4.   // 創建響應式對象 
  5.   return createReactiveObject(target); 
  6. function isObject(target) { 
  7.   return typeof target === "object" && target !== null
  8. function hasOwn(target,key){ 
  9.   return target.hasOwnProperty(key); 
  10. function createReactiveObject(target) { 
  11.   if (!isObject(target)) { 
  12.     return target; 
  13.   } 
  14.   let observed = toProxy.get(target); 
  15.   if(observed){ // 判斷是否被代理過 
  16.     return observed; 
  17.   } 
  18.   if(toRaw.has(target)){ // 判斷是否要重復代理 
  19.     return target; 
  20.   } 
  21.   const handlers = { 
  22.     get(target, key, receiver) { 
  23.         let res = Reflect.get(target, key, receiver); 
  24.         track(target,'get',key); // 依賴收集== 
  25.         return isObject(res)  
  26.         ?reactive(res):res; 
  27.     }, 
  28.     set(target, key, value, receiver) { 
  29.         let oldValue = target[key]; 
  30.         let hadKey = hasOwn(target,key); 
  31.         let result = Reflect.set(target, key, value, receiver); 
  32.         if(!hadKey){ 
  33.           trigger(target,'add',key); // 觸發添加 
  34.         }else if(oldValue !== value){ 
  35.           trigger(target,'set',key); // 觸發修改 
  36.         } 
  37.         return result; 
  38.     }, 
  39.     deleteProperty(target, key) { 
  40.       console.log("刪除"); 
  41.       const result = Reflect.deleteProperty(target, key); 
  42.       return result; 
  43.     } 
  44.   }; 
  45.    
  46.   // 開始代理 
  47.   observed = new Proxy(target, handlers); 
  48.   toProxy.set(target,observed); 
  49.   toRaw.set(observed,target); // 做映射表 
  50.   return observed; 

總結

Proxy 相比于 defineProperty 的優勢:

基于 Proxy 和 Reflect ,可以原生監聽數組,可以監聽對象屬性的添加和刪除

不需要深度遍歷監聽:判斷當前 Reflect.get 的返回值是否為 Object ,如果是則再通過 reactive 方法做代理, 這樣就實現了深度觀測

只在 getter 時才對對象的下一層進行劫持(優化了性能)

所以,建議使用 Proxy 監測變量變化

參考

MDN

 

帶你了解 vue-next(Vue 3.0)之 爐火純青

 

責任編輯:武曉燕 來源: 三分鐘學前端
相關推薦

2022-05-13 08:12:00

JMeter測試計劃

2023-08-27 21:25:20

用戶畫像數據數據分析

2024-08-28 08:38:51

2010-11-17 15:43:55

軟件測試Bug

2018-06-25 08:33:33

技術總監職場規劃

2018-12-13 11:52:48

2012-05-24 14:58:55

開源代碼

2022-03-10 11:25:51

InnoDB優化

2023-09-27 22:44:18

數據遷移數據庫

2022-04-26 06:36:09

渠道分析數據采集

2023-08-28 08:52:49

監聽頁面用戶

2019-10-15 09:20:40

Linux系統服務器

2020-08-03 08:30:00

JSCSS排序

2016-09-21 10:18:26

阿里Dubbo性能測試

2025-09-03 02:46:00

Vue.js響應式變量

2011-03-11 09:53:46

FacebookMySQL

2021-05-05 10:48:33

滲透測試漏洞網絡攻擊

2021-08-19 09:00:12

監控文件Python

2017-07-20 13:11:46

Code ReviewPR評審

2024-04-10 10:15:16

監聽
點贊
收藏

51CTO技術棧公眾號

久久久美女毛片| 郴州新闻综合频道在线直播| 欧美视频在线播放| 亚洲视频专区在线| 三级外国片在线观看视频| 99在线精品一区二区三区| 久久久久九九九| 久久久成人网| 日本午夜精品一区二区三区| 亚洲专区一区二区三区| 久草一本av| 午夜精品一区二区三区在线播放 | 91成人精品观看| 亚洲一区二区精品久久av| www.99av| 欧美一卡2卡三卡4卡5免费| 在线欧美激情| 国产精品久久久久av福利动漫| 粉嫩嫩av羞羞动漫久久久| 超碰在线电影| 久久精品国产69国产精品亚洲| 成人综合一区| 中文字幕色呦呦| 7777精品伊人久久久大香线蕉的| 视频在线亚洲| 中文字幕不卡每日更新1区2区| 亚洲激情自拍偷拍| 涩涩涩久久久成人精品| 国产精品久久久久久久久久久久午夜片 | 在线观看一区日韩| 在线播放一区二区精品视频| 日本高清不卡三区| 亚洲成av人影院| vam成人资源在线观看| 图片区小说区区亚洲五月| 激情成人中文字幕| 欧美在线关看| 色总=综合色| 夜夜躁狠狠躁日日躁2021日韩| 狠狠干视频网站| 亚洲激情久久久| 秋霞国产午夜精品免费视频| 蜜桃免费在线| 成人h视频在线观看| 亚洲男女一区二区三区| 香蕉成人影院| 久久精品无码中文字幕| 国产一区二区精品丝袜| 懂色av一区二区在线播放| 尤物在线网址| 美女黄色片网站| 日韩不卡在线观看| 久久成人av少妇免费| 美女的胸无遮挡在线观看| 亚洲精品日韩在线观看| 亚洲国产另类久久精品| 国产精品一区二区视频| 国内激情视频在线观看| 成人毛片100部免费看| 日韩大陆欧美高清视频区| 久久午夜精品| 六九午夜精品视频| 日本免费专区| 久久视频在线观看中文字幕| 欧美一区二区三区色| 免费在线看成人av| 日韩精品中文字幕吗一区二区| 亚洲视频第二页| 91在线播放国产| 亚洲精品一区二区在线观看| 成人精品一区二区三区中文字幕| 成人动漫视频在线观看| 九色porn| 亚洲一区二区三区午夜| 欧美黑人性视频| 日韩欧美在线视频日韩欧美在线视频| 日韩成人一区二区| 北条麻妃一区二区三区在线观看 | 欧美一区二区啪啪| 亚洲男人天堂一区| 视频一区二区国产| 999在线精品| 在线免费观看的av| 日韩一二在线观看| 国产成人免费xxxxxxxx| 国产成人精品三级高清久久91| 毛片av在线| 2023欧美最顶级a∨艳星| 伊人久久大香线蕉午夜av| 热门国产精品亚洲第一区在线| 日韩一二三区视频| 色综合天天在线| 国产精品久久久久久亚洲伦 | 成人性生活视频免费看| 成人女保姆的销魂服务| 丝袜一区二区三区| 在线电影院国产精品| 欧美性生交xxxxxdddd| 中文字幕亚洲精品在线观看| 国产成人综合精品三级| 午夜在线精品偷拍| 在线播放一区| 99久久99热这里只有精品| 亚洲综合影院| 日韩av懂色| 久久精品97| 91精品在线免费视频| 91tv亚洲精品香蕉国产一区| 久草成色在线| 俺来俺也去www色在线观看| av午夜在线| 一区二区三区视频网站| 免费福利在线视频| 亚洲字幕成人中文在线观看| 日韩视频国产视频| 精品国精品国产| 国产视频在线观看一区二区| 日韩视频永久免费| 日韩av影视综合网| 久久人人爽人人爽人人片亚洲| 久久精品国产v日韩v亚洲| 亚洲一区www| 在线播放亚洲激情| 日韩性生活视频| 日韩女优在线播放| 51国偷自产一区二区三区| 9a蜜桃久久久久久免费| 日韩wuma| 成人在线观看a| 国产网友自拍电影在线| 蜜桃专区在线| 成人在线高清免费| 性欧美video另类hd尤物| 亚洲一二三四| 视频小说一区二区| 日本色综合中文字幕| 91日韩在线专区| 午夜精品一区二区三区免费视频 | 日韩经典av| 国产精品视频一区二区三区四蜜臂| 成人看的视频| 精品一区二区免费视频| 最新国产の精品合集bt伙计| 欧美日本精品一区二区三区| 最新69国产成人精品视频免费 | 91碰在线视频| 欧美午夜一区二区| 亚洲性无码av在线| 成人h视频在线观看播放| 亚洲午夜精品一区二区| 最近中文字幕2019第二页视频| av在线1区2区| 中文字幕精品影院| 自由日本语热亚洲人| 日韩精品一区二区久久| 成人一区二区三区视频| 在线中文字幕一区二区| 91精品国产91久久久久| 日韩av一级大片| 一色桃子在线| 成人激情自拍| 成人av免费在线播放| 精品国精品国产| 成人午夜一级二级三级| 欧美福利电影网| 国产精品福利在线观看| 国模吧无码一区二区三区| 久久人体大尺度| 国产一区二三区| 日韩午夜激情视频| 国产高清精品一区| 日本综合在线| 综合色一区二区| 欧美午夜片在线看| yellow视频在线观看一区二区| 四色成人av永久网址| 久久人人99| 色综合久久中文字幕| 国产日本欧美一区二区三区在线| 777.av| 欧美日韩激情| 亚洲地区一二三色| 成人免费福利视频| 亚洲做受高潮| 欧美私人啪啪vps| 欧美性猛片xxxx免费看久爱| 国产高清视频一区三区| 国产小视频精品| 9l视频自拍蝌蚪9l视频成人| 国产凹凸在线观看一区二区| 精品国产乱码久久久久久夜甘婷婷 | 精品精品国产国产自在线| 亚洲精品一区二| 麻豆视频在线观看免费网站黄| 国内揄拍国内精品久久| 欧美日韩一级视频| 麻豆久久久9性大片| 国产成人精品日本亚洲| 国外亚洲成av人片在线观看| 你懂的国产精品|