JavaScript中的數(shù)據(jù)劫持和數(shù)據(jù)代理使用
一、數(shù)據(jù)劫持(Data Hijacking)
1. 生活中的例子
假設(shè)你開(kāi)了一家餐廳,你雇了一個(gè) “賬房先生”(收銀員),他的工作是:
- 任何客人 查詢賬單 時(shí),他都會(huì)記錄查詢行為,并告訴客人正確的金額。
- 任何客人 修改賬單 時(shí),他都會(huì)檢查金額是否合理,并記錄修改情況。
現(xiàn)實(shí)中的“賬房先生”就是 Object.defineProperty(),它攔截每個(gè)數(shù)據(jù)的讀取和修改。
2. 代碼示例
我們用 Object.defineProperty() 來(lái)模擬這個(gè) “賬房先生”:
let bill = {}; // 創(chuàng)建賬單對(duì)象
Object.defineProperty(bill, 'amount', {
get() {
console.log('客人查詢了賬單金額');
return 100; // 假設(shè)賬單金額是100
},
set(value) {
console.log(`客人修改了賬單金額為 ${value}`);
}
});
console.log(bill.amount); // 觸發(fā) get 方法,查詢金額
bill.amount = 200; // 觸發(fā) set 方法,修改金額運(yùn)行結(jié)果:
客人查詢了賬單金額
100
客人修改了賬單金額為 200
問(wèn)題:
- 這個(gè)方法 只能劫持已
sh()、pop()這樣的操作不會(huì)觸發(fā)set。 - 必須對(duì)每個(gè)屬性單獨(dú)定義
get/set,如 - 有的屬性,如果
bill里沒(méi)有amount,defineProperty無(wú)法監(jiān)聽(tīng)到新增的amount屬性。 - 無(wú)法監(jiān)聽(tīng)數(shù)組變化,比如
pu - 果對(duì)象有很多屬性,就要定義很多次,性能較低。
二、數(shù)據(jù)代理(Data Proxy)
1. 生活中的例子
假設(shè)你家有一個(gè) 智能管家(類似于小愛(ài)同學(xué)、Siri),你可以問(wèn)他:
- "今天天氣怎么樣?" 他會(huì)去查天氣然后告訴你(相當(dāng)于攔截 讀取 操作)。
- "幫我把燈光調(diào)亮一些" 他會(huì)控制家里的燈光調(diào)亮(相當(dāng)于攔截 修改 操作)。
智能管家就是 Proxy,它可以代理整個(gè)房子里所有的設(shè)備,而不需要逐個(gè)綁定(不像 Object.defineProperty() 必須單獨(dú)劫持每個(gè)設(shè)備的控制)。
2. 代碼示例
用 Proxy 實(shí)現(xiàn)智能管家的功能:
let house = { temperature: 22, light: 'off' }; // 房子里的設(shè)備
let smartHouse = new Proxy(house, {
get(target, key) {
console.log(`查詢 ${key} 的狀態(tài):${target[key]}`);
return target[key];
},
set(target, key, value) {
console.log(`修改 ${key} 的狀態(tài)為 ${value}`);
target[key] = value;
return true;
}
});
console.log(smartHouse.temperature); // 觸發(fā) get,查詢溫度
smartHouse.light = 'on'; // 觸發(fā) set,修改燈光狀態(tài)運(yùn)行結(jié)果:
查詢 temperature 的狀態(tài):22
修改 light 的狀態(tài)為 on
Proxy 的優(yōu)勢(shì):
- 可以代理整個(gè)對(duì)象,不用為每個(gè)屬性單獨(dú)定義
get/set。 - 支持監(jiān)聽(tīng)新增屬性,比如
house.newDevice = 'TV'也能被攔截! - 支持?jǐn)?shù)組,例如監(jiān)聽(tīng)
push()、pop()這些操作。
三、對(duì)比總結(jié)
| 對(duì)比項(xiàng) | 數(shù)據(jù)劫持(Object.defineProperty) | 數(shù)據(jù)代理(Proxy) |
|---|---|---|
| Vue 版本 | Vue 2.x | Vue 3.x |
| 監(jiān)聽(tīng)對(duì)象 | 只能監(jiān)聽(tīng)已有的屬性 | 可以監(jiān)聽(tīng)整個(gè)對(duì)象 |
| 監(jiān)聽(tīng)新增/刪除屬性 | 不能監(jiān)聽(tīng)(需 Vue.set) | 可直接監(jiān)聽(tīng) |
| 監(jiān)聽(tīng)數(shù)組 | 需要重寫(xiě)數(shù)組方法 | 原生支持 |
| 深層嵌套 | 需要遞歸遍歷 | 訪問(wèn)時(shí)自動(dòng)代理 |
| 兼容性 | ES5 及以上 | 僅支持 ES6 及以上 |
四、再舉一個(gè)更貼近開(kāi)發(fā)的例子
使用 Object.defineProperty() 監(jiān)聽(tīng)對(duì)象
let person = { name: 'Alice', age: 25 };
function makeReactive(obj) {
for (let key in obj) {
let value = obj[key];
Object.defineProperty(obj, key, {
get() {
console.log(`讀取 ${key}: ${value}`);
return value;
},
set(newVal) {
console.log(`修改 ${key} 為 ${newVal}`);
value = newVal;
}
});
}
}
makeReactive(person);
console.log(person.name); // 讀取 name
person.age = 30; // 修改 age問(wèn)題:
- 不能監(jiān)聽(tīng)
person.newProp = 'test'這樣的新增屬性。 - 不能監(jiān)聽(tīng)
delete person.age這樣的刪除操作。
使用 Proxy 監(jiān)聽(tīng)整個(gè)對(duì)象
let person = { name: 'Alice', age: 25 };
let reactivePerson = new Proxy(person, {
get(target, key) {
console.log(`讀取 ${key}: ${target[key]}`);
return target[key];
},
set(target, key, value) {
console.log(`修改 ${key} 為 ${value}`);
target[key] = value;
return true;
},
deleteProperty(target, key) {
console.log(`刪除屬性 ${key}`);
return delete target[key];
}
});
console.log(reactivePerson.name); // 讀取 name
reactivePerson.age = 30; // 修改 age
reactivePerson.newProp = 'test'; // 監(jiān)聽(tīng)新增屬性
delete reactivePerson.age; // 監(jiān)聽(tīng)刪除屬性優(yōu)勢(shì):
Proxy可以監(jiān)聽(tīng)對(duì)象的新增/刪除屬性,而Object.defineProperty()不行!Proxy可以監(jiān)聽(tīng)數(shù)組的變化,而Object.defineProperty()不能監(jiān)聽(tīng)push()、splice()。Proxy可以代理整個(gè)對(duì)象,而Object.defineProperty()需要遍歷所有屬性,性能較差。
五、總結(jié)
| 概念 | 方式 | 適用場(chǎng)景 | 優(yōu)勢(shì) |
|---|---|---|---|
| 數(shù)據(jù)劫持 | Object.defineProperty() | Vue 2.x | 兼容性好,支持老瀏覽器,但不能監(jiān)聽(tīng)新增/刪除屬性 |
| 數(shù)據(jù)代理 | Proxy | Vue 3.x | 功能更強(qiáng),可監(jiān)聽(tīng)新增/刪除,支持?jǐn)?shù)組操作 |
結(jié)論:
- 如果是 Vue 2,只能用
Object.defineProperty()來(lái)實(shí)現(xiàn)數(shù)據(jù)劫持。 - 如果是 Vue 3,推薦用
Proxy,因?yàn)樗鼜?qiáng)大,能處理所有數(shù)據(jù)變化。
以上為個(gè)人經(jīng)驗(yàn),希望能給大家一個(gè)參考,也希望大家多多支持腳本之家。
相關(guān)文章
解決 viewer.js 動(dòng)態(tài)更新圖片導(dǎo)致無(wú)法預(yù)覽的問(wèn)題
Viewer.js 是一款強(qiáng)大的圖片查看器,這篇文章主要介紹了解決 viewer.js 動(dòng)態(tài)更新圖片導(dǎo)致無(wú)法預(yù)覽的問(wèn)題 ,需要的朋友可以參考下2019-05-05
JS實(shí)現(xiàn)的模仿QQ頭像資料卡顯示與隱藏效果
這篇文章主要介紹了JS實(shí)現(xiàn)的模仿QQ頭像資料卡顯示與隱藏效果,涉及javascript鼠標(biāo)事件響應(yīng)及頁(yè)面元素屬性動(dòng)態(tài)操作相關(guān)技巧,需要的朋友可以參考下2017-04-04
JavaScript實(shí)現(xiàn)簡(jiǎn)易登錄注冊(cè)頁(yè)面
這篇文章主要為大家詳細(xì)介紹了JavaScript實(shí)現(xiàn)簡(jiǎn)易登錄注冊(cè)頁(yè)面,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下<BR>2022-01-01

