国产99久久精品_欧美日本韩国一区二区_激情小说综合网_欧美一级二级视频_午夜av电影_日本久久精品视频

最新文章專題視頻專題問答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
主站蜘蛛池模板: 国产视频一区二区 | 久久久高清国产999尤物 | 久久久久成人精品一区二区 | 国产成人精品亚洲一区 | 高清一区二区亚洲欧美日韩 | 国产成人在线观看免费网站 | 日本不卡视频一区二区三区 | 精品免费久久久久国产一区 | 国产一区91 | 永久免费观看的毛片的网站下载 | 日韩二区三区 | 图片区 日韩 欧美 亚洲 | 欧美日韩亚洲无线码在线观看 | 欧美综合自拍亚洲综合 | 久久久久久亚洲精品中文字幕 | 92国产福利久久青青草原 | 五月天中文字幕 | 亚洲一区二区免费 | 国产高清视频免费 | 久久a毛片 | 欧美精品第1页在线播放 | 91精品国产乱码久久久久久 | 亚洲精品免费观看 | 亚洲十欧美十日韩十国产 | 亚洲 欧美 日韩 在线 | 日韩成人在线电影 | 日韩精品欧美高清区 | 精品久久久久久久一区二区手机版 | 亚洲欧美日韩精品高清 | 91欧美激情一区二区三区成人 | 伊人激情网 | 日韩国产欧美一区二区三区在线 | 老司机精品视频一区二区 | 日韩电影免费在线观看网址 | 日韩欧美一区二区三区久久 | 国产资源免费 | 国产成人精品第一区二区 | 国产精品激情综合久久 | 国产精品…在线观看 | 国产区精品视频 | 一区二区高清在线 |