JavaScript中的Proxy對象
Js中Proxy對象
Proxy對象用于定義基本操作的自定義行為,例如屬性查找、賦值、枚舉、函數(shù)調(diào)用等。
語法
const proxy = new Proxy(target, handler);
- target: 要使用Proxy包裝的目標(biāo)對象,可以是任何類型的對象,包括原生數(shù)組,函數(shù),甚至另一個代理。
- handler: 一個通常以函數(shù)作為屬性的對象,各屬性中的函數(shù)分別定義了在執(zhí)行各種操作時代理proxy的行為。
描述
Proxy用于修改某些操作的默認(rèn)行為,也可以理解為在目標(biāo)對象之前架設(shè)一層攔截,外部所有的訪問都必須先通過這層攔截,因此提供了一種機制,可以對外部的訪問進行過濾和修改。這個詞的原理為代理,在這里可以表示由它來代理某些操作,譯為代理器。
var target = {a: 1};
var proxy = new Proxy(target, {
set: function(target, key, value, receiver){
console.log("watch");
return Reflect.set(target, key, value, receiver);
},
get: function(target, key, receiver){
return target[key];
}
});
proxy.a = 11; // watch
console.log(target); // {a: 11}
Object.defineProperty是用于監(jiān)聽屬性,而Proxy是監(jiān)聽整個對象,通過調(diào)用new Proxy(),可以創(chuàng)建一個代理用來替代另一個對象被稱為目標(biāo),這個代理對目標(biāo)對象進行了虛擬,因此該代理與該目標(biāo)對象表面上可以被當(dāng)作同一個對象來對待。代理允許攔截在目標(biāo)對象上的底層操作,而這原本是Js引擎的內(nèi)部能力,攔截行為使用了一個能夠響應(yīng)特定操作的函數(shù),即通過Proxy去對一個對象進行代理之后,我們將得到一個和被代理對象幾乎完全一樣的對象,并且可以從底層實現(xiàn)對這個對象進行完全的監(jiān)控。
// 常見的一道面試題 實現(xiàn) a===1&&a===2&&a===3 為true
// Object.defineProperty 定義的是屬性
// 可以實現(xiàn)對于題目的要求
var _a = 0;
Object.defineProperty(window, "a", {
get:function(){
return ++_a;
}
})
console.log(a===1 && a===2 && a===3); // true
// proxy 代理的是對象
// 因此在調(diào)用時實際與題目要求并不太相符
// 但同樣也是一種實現(xiàn)方式
var _a = 0;
var proxy = new Proxy(window, {
set: function(target, key, value, receiver){
return Reflect.set(target, key, value, receiver);
},
get: function(target, key, receiver){
if(key === "a") return ++_a;
else return window[key];
}
});
console.log(proxy.a===1 && proxy.a===2 && proxy.a===3); //true
方法
Proxy.revocable()
Proxy.revocable(target, handler)
Proxy.revocable()方法可以用來創(chuàng)建一個可撤銷的代理對象,其返回一個包含了代理對象本身和它的撤銷方法的可撤銷Proxy對象。
- target: 將用Proxy封裝的目標(biāo)對象,可以是任何類型的對象,包括原生數(shù)組,函數(shù),甚至可以是另外一個代理對象。
- handler: 一個對象,其屬性是一批可選的函數(shù),這些函數(shù)定義了對應(yīng)的操作被執(zhí)行時代理的行為。
該方法的返回值是一個對象,其結(jié)構(gòu)為{"proxy": proxy, "revoke": revoke},一旦某個代理對象被撤銷,它將變得幾乎完全不可調(diào)用,在它身上執(zhí)行任何的可代理操作都會拋出TypeError異常,注意可代理操作一共有14種,執(zhí)行這14種操作以外的操作不會拋出異常。一旦被撤銷,這個代理對象便不可能被直接恢復(fù)到原來的狀態(tài),同時和它關(guān)聯(lián)的目標(biāo)對象以及處理器對象都有可能被垃圾回收掉。再次調(diào)用撤銷方法revoke()則不會有任何效果,但也不會報錯。
var revocable = Proxy.revocable({}, {
get: function(target, key) {
return `[[ ${key} ]]`;
}
});
var proxy = revocable.proxy;
console.log(proxy.example); // [[ example ]]
revocable.revoke();
// console.log(proxy.example); // 拋出 TypeError
// proxy.example = 1; // 拋出 TypeError
// delete proxy.example; // 拋出 TypeError
// typeof proxy // "object",因為 typeof 不屬于可代理操作
handler對象方法
handler對象是一個容納一批特定屬性的占位符對象,它包含有Proxy的各個捕獲器trap。所有的捕捉器是可選的,如果沒有定義某個捕捉器,那么就會保留源對象的默認(rèn)行為。
- handler.getPrototypeOf(): Object.getPrototypeOf方法的捕捉器。
- handler.setPrototypeOf(): Object.setPrototypeOf方法的捕捉器。
- handler.isExtensible(): Object.isExtensible方法的捕捉器。
- handler.preventExtensions(): Object.preventExtensions方法的捕捉器。
- handler.getOwnPropertyDescriptor(): Object.getOwnPropertyDescriptor方法的捕捉器。
- handler.defineProperty(): Object.defineProperty方法的捕捉器。
- handler.has(): in操作符的捕捉器。
- handler.get(): 屬性讀取操作的捕捉器。
- handler.set(): 屬性設(shè)置操作的捕捉器。
- handler.deleteProperty(): delete操作符的捕捉器。
- handler.ownKeys(): Reflect.ownKeys、Object.getOwnPropertyNames、Object.keys、Object.getOwnPropertySymbols方法的捕捉器。
- handler.apply(): 函數(shù)調(diào)用操作的捕捉器。
- handler.construct(): new操作符的捕捉器。
var target = {
a: 1,
f: function(...args){
console.log(...args);
}
};
var proxy = new Proxy(target, {
getPrototypeOf: function(target) {
console.log("getPrototypeOf");
return Object.getPrototypeOf(target);
},
setPrototypeOf: function(target, prototype) {
console.log("setPrototypeOf");
return Object.setPrototypeOf(target, prototype);
},
isExtensible: function(target) {
console.log("isExtensible");
return Object.isExtensible(target);
},
preventExtensions: function(target) {
console.log("preventExtensions");
return Object.preventExtensions(target);
},
getOwnPropertyDescriptor: function(target, prop) {
console.log("getOwnPropertyDescriptor");
return Object.getOwnPropertyDescriptor(target, prop);
},
defineProperty: function(target, prop, descriptor) {
console.log("defineProperty");
return Object.defineProperty(target, prop, descriptor);
},
has: function(target, prop) {
console.log("has");
return prop in target;
},
get: function(target, prop, receiver) {
console.log("get");
return target[prop];
},
set: function(target, prop, value, receiver) {
console.log("set");
target[prop] = value;
return true;
},
deleteProperty: function(target, property) {
console.log("deleteProperty");
delete target[property];
return true;
},
ownKeys: function(target) {
console.log("ownKeys");
return Reflect.ownKeys(target);
}
})
var proxyF = new Proxy(target.f, {
construct: function(target, argumentsList, newTarget) {
console.log("construct");
return new target(...argumentsList);
},
apply: function(target, thisArg, argumentsList) {
console.log("apply");
return target.apply(thisArg, argumentsList);
},
})
const _prototype = {test: 1};
Object.setPrototypeOf(proxy, _prototype); // setPrototypeOf
console.log(Object.getPrototypeOf(proxy)); // getPrototypeOf // { test: 1 }
Object.preventExtensions(proxy); // preventExtensions
console.log(Object.isExtensible(proxy)); // isExtensible // false
Object.defineProperty(proxy, "a", {configurable: true}); // defineProperty
console.log(Object.getOwnPropertyDescriptor(proxy, "a")); // getOwnPropertyDescriptor // { value: 1, writable: true, enumerable: true, configurable: true }
proxy.a = 11; // set
console.log(proxy.a); // get // 11
console.log(Object.keys(proxy)); // ownKeys getOwnPropertyDescriptor getOwnPropertyDescriptor // [ 'a', 'f' ]
delete proxy.a; // deleteProperty
console.log("a" in proxy); // has // false
proxyF(1, 2, 3); // apply 1 2 3
new proxyF(1, 2, 3); // construct 1 2 3
每日一題
https://github.com/WindrunnerMax/EveryDay
以上就是JavaScript中的Proxy對象的詳細內(nèi)容,更多關(guān)于JavaScript Proxy對象的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
JavaScript實現(xiàn)維吉尼亞(Vigenere)密碼算法實例
Vigenere密碼就是一種傳統(tǒng)加密技術(shù),它是多表代換密碼,能夠有效改進單表代換密碼的詞頻分布特征問題,本文用JavaScript實現(xiàn)維吉尼亞(Vigenere)密碼算法2013-11-11
Bootstrap彈出帶合法性檢查的登錄框?qū)嵗a【推薦】
這篇文章主要介紹了Bootstrap彈出帶合法性檢查的登錄框?qū)嵗a【推薦】的相關(guān)資料,需要的朋友可以參考下2016-06-06
在 javascript 中如何快速獲取數(shù)組指定位置的元素
這篇文章主要介紹了在 javascript 中快速獲取數(shù)組指定位置的元素,本文通過實例代碼給大家介紹的非常詳細,對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-04-04
控制臺報錯:Cannot?access?'xxx'?before?initializatio
這篇文章主要給大家介紹了關(guān)于控制臺報錯:Cannot?access?'xxx'?before?initialization的解決方法,文中通過代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考借鑒價值,需要的朋友可以參考下2023-11-11
Javascript獲取當(dāng)前時間函數(shù)和時間操作小結(jié)
這篇文章主要介紹了Javascript獲取當(dāng)前時間函數(shù)和時間操作小結(jié),本文根據(jù)項目實際需求總結(jié)而來,需要的朋友可以參考下2014-10-10
unapp微信小程序轉(zhuǎn)發(fā)分享及攜帶參數(shù)的2種方式
這篇文章主要給大家介紹了關(guān)于unapp微信小程序轉(zhuǎn)發(fā)分享及攜帶參數(shù)的2種方式,我們在微信小程序中經(jīng)常會使用到分享商品海報,或者是重繪微信小程序分享鏈的圖片功能,需要的朋友可以參考下2023-11-11
js實現(xiàn)二級菜單點擊顯示當(dāng)前內(nèi)容效果
這篇文章主要介紹了js實現(xiàn)二級菜單點擊顯示當(dāng)前內(nèi)容效果,具有一定的參考價值,感興趣的小伙伴們可以參考一下2018-04-04

