• <fieldset id="8imwq"><menu id="8imwq"></menu></fieldset>
  • <bdo id="8imwq"><input id="8imwq"></input></bdo>
    最新文章專題視頻專題問答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
    主站蜘蛛池模板: 337P亚洲精品色噜噜| 国产高清国内精品福利99久久| 久久99国产精品久久99| 亚洲一区二区三区国产精品| 999国产精品色在线播放| 四虎影视永久在线观看精品| 久久精品成人一区二区三区| 国内精品久久久久久野外| 亚洲av无码精品网站| 精品久久久久中文字| 久久精品国产一区二区三区日韩| 精品综合久久久久久888蜜芽| 无码国产亚洲日韩国精品视频一区二区三区| 88国产精品欧美一区二区三区| 国产精品无码av在线播放| 亚洲精品成人片在线观看精品字幕| 久久久久人妻一区精品果冻| 91在线手机精品超级观看| 久久精品成人免费网站| 国产a∨精品一区二区三区不卡| 亚洲精品~无码抽插| 亚洲午夜成人精品电影在线观看| 精品成人av一区二区三区| 国产精品乱伦| 国产高清在线精品一本大道国产| 久久亚洲国产精品一区二区| 国产精品久久久久久搜索| 91精品视频网站| 996久久国产精品线观看| 国产成人精品一区二区三区免费| 精品熟女少妇av免费久久| 人妻少妇精品视频二区| 无码aⅴ精品一区二区三区浪潮| 中文字幕亚洲精品无码| 亚洲国产精品无码成人片久久| 亚洲国产精品一区二区成人片国内| 中文字幕精品一区二区三区视频| 日韩人妻精品无码一区二区三区| 人妻少妇精品视中文字幕国语| 精品久久久久久亚洲精品| 成人伊人精品色XXXX视频|