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-05JS實(shí)現(xiàn)的模仿QQ頭像資料卡顯示與隱藏效果
這篇文章主要介紹了JS實(shí)現(xiàn)的模仿QQ頭像資料卡顯示與隱藏效果,涉及javascript鼠標(biāo)事件響應(yīng)及頁(yè)面元素屬性動(dòng)態(tài)操作相關(guān)技巧,需要的朋友可以參考下2017-04-04JavaScript實(shí)現(xiàn)簡(jiǎn)易登錄注冊(cè)頁(yè)面
這篇文章主要為大家詳細(xì)介紹了JavaScript實(shí)現(xiàn)簡(jiǎn)易登錄注冊(cè)頁(yè)面,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下<BR>2022-01-01