<span id="mktg5"></span>

<i id="mktg5"><meter id="mktg5"></meter></i>

        <label id="mktg5"><meter id="mktg5"></meter></label>
        最新文章專題視頻專題問答1問答10問答100問答1000問答2000關鍵字專題1關鍵字專題50關鍵字專題500關鍵字專題1500TAG最新視頻文章推薦1 推薦3 推薦5 推薦7 推薦9 推薦11 推薦13 推薦15 推薦17 推薦19 推薦21 推薦23 推薦25 推薦27 推薦29 推薦31 推薦33 推薦35 推薦37視頻文章20視頻文章30視頻文章40視頻文章50視頻文章60 視頻文章70視頻文章80視頻文章90視頻文章100視頻文章120視頻文章140 視頻2關鍵字專題關鍵字專題tag2tag3文章專題文章專題2文章索引1文章索引2文章索引3文章索引4文章索引5123456789101112131415文章專題3
        問答文章1 問答文章501 問答文章1001 問答文章1501 問答文章2001 問答文章2501 問答文章3001 問答文章3501 問答文章4001 問答文章4501 問答文章5001 問答文章5501 問答文章6001 問答文章6501 問答文章7001 問答文章7501 問答文章8001 問答文章8501 問答文章9001 問答文章9501
        當前位置: 首頁 - 科技 - 知識百科 - 正文

        實例解析ES6 Proxy使用場景介紹

        來源:懂視網 責編:小采 時間:2020-11-27 22:22:03
        文檔

        實例解析ES6 Proxy使用場景介紹

        實例解析ES6 Proxy使用場景介紹:ES6 中的箭頭函數、數組解構、rest 參數等特性一經實現就廣為流傳,但類似 Proxy 這樣的特性卻很少見到有開發者在使用,一方面在于瀏覽器的兼容性,另一方面也在于要想發揮這些特性的優勢需要開發者深入地理解其使用場景。就我個人而言是非常喜歡 ES6 的
        推薦度:
        導讀實例解析ES6 Proxy使用場景介紹:ES6 中的箭頭函數、數組解構、rest 參數等特性一經實現就廣為流傳,但類似 Proxy 這樣的特性卻很少見到有開發者在使用,一方面在于瀏覽器的兼容性,另一方面也在于要想發揮這些特性的優勢需要開發者深入地理解其使用場景。就我個人而言是非常喜歡 ES6 的

        ES6 中的箭頭函數、數組解構、rest 參數等特性一經實現就廣為流傳,但類似 Proxy 這樣的特性卻很少見到有開發者在使用,一方面在于瀏覽器的兼容性,另一方面也在于要想發揮這些特性的優勢需要開發者深入地理解其使用場景。就我個人而言是非常喜歡 ES6 的 Proxy,因為它讓我們以簡潔易懂的方式控制了外部對對象的訪問。在下文中,首先我會介紹 Proxy 的使用方式,然后列舉具體實例解釋 Proxy 的使用場景。

        Proxy,見名知意,其功能非常類似于設計模式中的代理模式,該模式常用于三個方面:

        1. 攔截和監視外部對對象的訪問
        2. 降低函數或類的復雜度
        3. 在復雜操作前對操作進行校驗或對所需資源進行管理

        在支持 Proxy 的瀏覽器環境中,Proxy 是一個全局對象,可以直接使用。Proxy(target, handler) 是一個構造函數,target 是被代理的對象,handlder 是聲明了各類代理操作的對象,最終返回一個代理對象。外界每次通過代理對象訪問 target 對象的屬性時,就會經過 handler 對象,從這個流程來看,代理對象很類似 middleware(中間件)。那么 Proxy 可以攔截什么操作呢?最常見的就是 get(讀?。?、set(修改)對象屬性等操作,完整的可攔截操作列表請點擊這里。此外,Proxy 對象還提供了一個 revoke 方法,可以隨時注銷所有的代理操作。在我們正式介紹 Proxy 之前,建議你對 Reflect 有一定的了解,它也是一個 ES6 新增的全局對象,詳細信息請參考MDN Reflect。

        Basic

        const target = { 
         name: 'Billy Bob',
         age: 15
        };
        
        const handler = { 
         get(target, key, proxy) {
         const today = new Date();
         console.log(`GET request made for ${key} at ${today}`);
        
         return Reflect.get(target, key, proxy);
         }
        };
        
        const proxy = new Proxy(target, handler);
        proxy.name;
        // => "GET request made for name at Thu Jul 21 2016 15:26:20 GMT+0800 (CST)"
        // => "Billy Bob"
        

        在上面的代碼中,我們首先定義了一個被代理的目標對象 target,然后聲明了包含所有代理操作的 handler 對象,接下來使用 Proxy(target, handler) 創建代理對象 proxy,此后所有使用 proxy 對 target 屬性的訪問都會經過 handler 的處理。

        1. 抽離校驗模塊

        讓我們從一個簡單的類型校驗開始做起,這個示例演示了如何使用 Proxy 保障數據類型的準確性:

        let numericDataStore = { 
         count: 0,
         amount: 1234,
         total: 14
        };
        
        numericDataStore = new Proxy(numericDataStore, { 
         set(target, key, value, proxy) {
         if (typeof value !== 'number') {
         throw Error("Properties in numericDataStore can only be numbers");
         }
         return Reflect.set(target, key, value, proxy);
         }
        });
        
        // 拋出錯誤,因為 "foo" 不是數值
        numericDataStore.count = "foo";
        
        // 賦值成功
        numericDataStore.count = 333;

        如果要直接為對象的所有屬性開發一個校驗器可能很快就會讓代碼結構變得臃腫,使用 Proxy 則可以將校驗器從核心邏輯分離出來自成一體:

        function createValidator(target, validator) { 
         return new Proxy(target, {
         _validator: validator,
         set(target, key, value, proxy) {
         if (target.hasOwnProperty(key)) {
         let validator = this._validator[key];
         if (!!validator(value)) {
         return Reflect.set(target, key, value, proxy);
         } else {
         throw Error(`Cannot set ${key} to ${value}. Invalid.`);
         }
         } else {
         throw Error(`${key} is not a valid property`)
         }
         }
         });
        }
        
        const personValidators = { 
         name(val) {
         return typeof val === 'string';
         },
         age(val) {
         return typeof age === 'number' && age > 18;
         }
        }
        class Person { 
         constructor(name, age) {
         this.name = name;
         this.age = age;
         return createValidator(this, personValidators);
         }
        }
        
        const bill = new Person('Bill', 25);
        
        // 以下操作都會報錯
        bill.name = 0; 
        bill.age = 'Bill'; 
        bill.age = 15; 

        通過校驗器和主邏輯的分離,你可以無限擴展 personValidators 校驗器的內容,而不會對相關的類或函數造成直接破壞。更復雜一點,我們還可以使用 Proxy 模擬類型檢查,檢查函數是否接收了類型和數量都正確的參數:

        let obj = { 
         pickyMethodOne: function(obj, str, num) { /* ... */ },
         pickyMethodTwo: function(num, obj) { /*... */ }
        };
        
        const argTypes = { 
         pickyMethodOne: ["object", "string", "number"],
         pickyMethodTwo: ["number", "object"]
        };
        
        obj = new Proxy(obj, { 
         get: function(target, key, proxy) {
         var value = target[key];
         return function(...args) {
         var checkArgs = argChecker(key, args, argTypes[key]);
         return Reflect.apply(value, target, args);
         };
         }
        });
        
        function argChecker(name, args, checkers) { 
         for (var idx = 0; idx < args.length; idx++) {
         var arg = args[idx];
         var type = checkers[idx];
         if (!arg || typeof arg !== type) {
         console.warn(`You are incorrectly implementing the signature of ${name}. Check param ${idx + 1}`);
         }
         }
        }
        
        obj.pickyMethodOne(); 
        // > You are incorrectly implementing the signature of pickyMethodOne. Check param 1
        // > You are incorrectly implementing the signature of pickyMethodOne. Check param 2
        // > You are incorrectly implementing the signature of pickyMethodOne. Check param 3
        
        obj.pickyMethodTwo("wopdopadoo", {}); 
        // > You are incorrectly implementing the signature of pickyMethodTwo. Check param 1
        
        // No warnings logged
        obj.pickyMethodOne({}, "a little string", 123); 
        obj.pickyMethodOne(123, {});
        
        

        2. 私有屬性

        在 JavaScript 或其他語言中,大家會約定俗成地在變量名之前添加下劃線 _ 來表明這是一個私有屬性(并不是真正的私有),但我們無法保證真的沒人會去訪問或修改它。在下面的代碼中,我們聲明了一個私有的 apiKey,便于 api 這個對象內部的方法調用,但不希望從外部也能夠訪問 api._apiKey:

        var api = { 
         _apiKey: '123abc456def',
         /* mock methods that use this._apiKey */
         getUsers: function(){}, 
         getUser: function(userId){}, 
         setUser: function(userId, config){}
        };
        
        // logs '123abc456def';
        console.log("An apiKey we want to keep private", api._apiKey);
        
        // get and mutate _apiKeys as desired
        var apiKey = api._apiKey; 
        api._apiKey = '987654321';
        
        

        很顯然,約定俗成是沒有束縛力的。使用 ES6 Proxy 我們就可以實現真實的私有變量了,下面針對不同的讀取方式演示兩個不同的私有化方法。第一種方法是使用 set / get 攔截讀寫請求并返回 undefined:

        let api = { 
         _apiKey: '123abc456def',
         getUsers: function(){ }, 
         getUser: function(userId){ }, 
         setUser: function(userId, config){ }
        };
        
        const RESTRICTED = ['_apiKey'];
        api = new Proxy(api, { 
         get(target, key, proxy) {
         if(RESTRICTED.indexOf(key) > -1) {
         throw Error(`${key} is restricted. Please see api documentation for further info.`);
         }
         return Reflect.get(target, key, proxy);
         },
         set(target, key, value, proxy) {
         if(RESTRICTED.indexOf(key) > -1) {
         throw Error(`${key} is restricted. Please see api documentation for further info.`);
         }
         return Reflect.get(target, key, value, proxy);
         }
        });
        
        // 以下操作都會拋出錯誤
        console.log(api._apiKey);
        api._apiKey = '987654321'; 

        第二種方法是使用 has 攔截 in 操作:

        var api = { 
         _apiKey: '123abc456def',
         getUsers: function(){ }, 
         getUser: function(userId){ }, 
         setUser: function(userId, config){ }
        };
        
        const RESTRICTED = ['_apiKey'];
        api = new Proxy(api, { 
         has(target, key) {
         return (RESTRICTED.indexOf(key) > -1) ?
         false :
         Reflect.has(target, key);
         }
        });
        
        // these log false, and `for in` iterators will ignore _apiKey
        console.log("_apiKey" in api);
        
        for (var key in api) { 
         if (api.hasOwnProperty(key) && key === "_apiKey") {
         console.log("This will never be logged because the proxy obscures _apiKey...")
         }
        }
        
        

        3. 訪問日志

        對于那些調用頻繁、運行緩慢或占用執行環境資源較多的屬性或接口,開發者會希望記錄它們的使用情況或性能表現,這個時候就可以使用 Proxy 充當中間件的角色,輕而易舉實現日志功能:

        let api = { 
         _apiKey: '123abc456def',
         getUsers: function() { /* ... */ },
         getUser: function(userId) { /* ... */ },
         setUser: function(userId, config) { /* ... */ }
        };
        
        function logMethodAsync(timestamp, method) { 
         setTimeout(function() {
         console.log(`${timestamp} - Logging ${method} request asynchronously.`);
         }, 0)
        }
        
        api = new Proxy(api, { 
         get: function(target, key, proxy) {
         var value = target[key];
         return function(...arguments) {
         logMethodAsync(new Date(), key);
         return Reflect.apply(value, target, arguments);
         };
         }
        });
        
        api.getUsers();
        
        

        4. 預警和攔截

        假設你不想讓其他開發者刪除 noDelete 屬性,還想讓調用 oldMethod 的開發者了解到這個方法已經被廢棄了,或者告訴開發者不要修改 doNotChange 屬性,那么就可以使用 Proxy 來實現:

        let dataStore = { 
         noDelete: 1235,
         oldMethod: function() {/*...*/ },
         doNotChange: "tried and true"
        };
        
        const NODELETE = ['noDelete']; 
        const NOCHANGE = ['doNotChange'];
        const DEPRECATED = ['oldMethod']; 
        
        dataStore = new Proxy(dataStore, { 
         set(target, key, value, proxy) {
         if (NOCHANGE.includes(key)) {
         throw Error(`Error! ${key} is immutable.`);
         }
         return Reflect.set(target, key, value, proxy);
         },
         deleteProperty(target, key) {
         if (NODELETE.includes(key)) {
         throw Error(`Error! ${key} cannot be deleted.`);
         }
         return Reflect.deleteProperty(target, key);
        
         },
         get(target, key, proxy) {
         if (DEPRECATED.includes(key)) {
         console.warn(`Warning! ${key} is deprecated.`);
         }
         var val = target[key];
        
         return typeof val === 'function' ?
         function(...args) {
         Reflect.apply(target[key], target, args);
         } :
         val;
         }
        });
        
        // these will throw errors or log warnings, respectively
        dataStore.doNotChange = "foo"; 
        delete dataStore.noDelete; 
        dataStore.oldMethod();

        5. 過濾操作

        某些操作會非常占用資源,比如傳輸大文件,這個時候如果文件已經在分塊發送了,就不需要在對新的請求作出相應(非絕對),這個時候就可以使用 Proxy 對當請求進行特征檢測,并根據特征過濾出哪些是不需要響應的,哪些是需要響應的。下面的代碼簡單演示了過濾特征的方式,并不是完整代碼,相信大家會理解其中的妙處:

        let obj = { 
         getGiantFile: function(fileId) {/*...*/ }
        };
        
        obj = new Proxy(obj, { 
         get(target, key, proxy) {
         return function(...args) {
         const id = args[0];
         let isEnroute = checkEnroute(id);
         let isDownloading = checkStatus(id); 
         let cached = getCached(id);
        
         if (isEnroute || isDownloading) {
         return false;
         }
         if (cached) {
         return cached;
         }
         return Reflect.apply(target[key], target, args);
         }
         }
        });
        
        

        6. 中斷代理

        Proxy 支持隨時取消對 target 的代理,這一操作常用于完全封閉對數據或接口的訪問。在下面的示例中,我們使用了 Proxy.revocable 方法創建了可撤銷代理的代理對象:

        let sensitiveData = { username: 'devbryce' };
        const {sensitiveData, revokeAccess} = Proxy.revocable(sensitiveData, handler);
        function handleSuspectedHack(){ 
         revokeAccess();
        }
        
        // logs 'devbryce'
        console.log(sensitiveData.username);
        handleSuspectedHack();
        // TypeError: Revoked
        console.log(sensitiveData.username);
        

        Decorator

        ES7 中實現的 Decorator,相當于設計模式中的裝飾器模式。如果簡單地區分 Proxy 和 Decorator 的使用場景,可以概括為:Proxy 的核心作用是控制外界對被代理者內部的訪問,Decorator 的核心作用是增強被裝飾者的功能。只要在它們核心的使用場景上做好區別,那么像是訪問日志這樣的功能,雖然本文使用了 Proxy 實現,但也可以使用 Decorator 實現,開發者可以根據項目的需求、團隊的規范、自己的偏好自由選擇。

        聲明:本網頁內容旨在傳播知識,若有侵權等問題請及時與本網聯系,我們將在第一時間刪除處理。TEL:177 7030 7066 E-MAIL:11247931@qq.com

        文檔

        實例解析ES6 Proxy使用場景介紹

        實例解析ES6 Proxy使用場景介紹:ES6 中的箭頭函數、數組解構、rest 參數等特性一經實現就廣為流傳,但類似 Proxy 這樣的特性卻很少見到有開發者在使用,一方面在于瀏覽器的兼容性,另一方面也在于要想發揮這些特性的優勢需要開發者深入地理解其使用場景。就我個人而言是非常喜歡 ES6 的
        推薦度:
        標簽: 介紹 pr 詳解
        • 熱門焦點

        最新推薦

        猜你喜歡

        熱門推薦

        專題
        Top
        主站蜘蛛池模板: 免费在线观看一区| 亚洲精品无码专区在线| 中文字幕免费观看全部电影| 国产色爽免费视频| 羞羞视频在线免费观看| 又粗又大又猛又爽免费视频| 无码色偷偷亚洲国内自拍| 国产成人aaa在线视频免费观看| 亚洲精品国产suv一区88| 国产免费69成人精品视频| 国产AV无码专区亚洲AV蜜芽| 国产一级淫片免费播放| www成人免费观看网站| 亚洲日本乱码在线观看| 中文字幕免费视频一| 亚洲综合校园春色| 国产在线观看免费完整版中文版| 国产精品亚洲二区在线| 2022中文字字幕久亚洲| 无码精品国产一区二区三区免费| 亚洲av鲁丝一区二区三区| 免费h片在线观看网址最新| 国产精品亚洲综合五月天| 免费一级毛片免费播放| 中文在线免费看视频| 亚洲欧洲精品久久| 老司机永久免费网站在线观看| 狼色精品人妻在线视频免费| 国产亚洲av片在线观看16女人| 亚洲精品免费在线视频| 亚洲av永久中文无码精品综合| 国产成人亚洲精品影院| 最近中文字幕大全免费视频 | 亚洲成?Ⅴ人在线观看无码| 成人片黄网站色大片免费观看cn| 午夜亚洲AV日韩AV无码大全| 成人免费福利电影| 日韩精品无码免费专区午夜 | 亚洲AV日韩综合一区| 亚洲av无码国产精品夜色午夜| 国产一卡二卡3卡四卡免费|