定義
??代理是一個中間者的角色,如生活中的中介,出于種種考慮/限制,一個物件不能直接訪問另一個物件,需要一個第三者(中間代理)牽線搭橋從而間接達到訪問目的,這樣的就是代理模式,
es6 中的代理
??es6 的 proxy 就是上面說的代理模式的實作,es6 幫我們在語法層面提供了這個新的api,讓我們可以很輕松的就使用了代理模式,
const p = new Proxy(target, handler)
target:要使用 Proxy 包裝的目標物件(可以是任何型別的物件,包括原生陣列,函式,甚至另一個代理)
handler:一個通常以函式作為屬性的物件
proxy 實體
const handler = {
get: function(obj, prop) {
return prop in obj ? obj[prop] : 37;
}
};
const p = new Proxy({}, handler);
p.a = 1;
p.b = undefined;
console.log(p.a, p.b); // 1, undefined
console.log('c' in p, p.c); // false, 37
應用實踐-模擬代理模式
??代理模式的應用非常常見,既可以是為了加強控制、拓展功能、提高性能,也可以僅僅是為了優化我們的代碼結構、實作功能的解耦,無論是出于什么目的,這種模式的套路就只有一個—— A 不能直接訪問 B,A 需要借助一個幫手來訪問 B,這個幫手就是代理器,需要這種代理器的就是代理模式的應用場景,
通常開發中最常見的代理型別:事件代理、虛擬代理、快取代理、保護代理;
- 事件代理:代理 DOM
- 虛擬代理:代理 DOM
- 快取代理:代理函式
- 保護代理:代理物件
事件代理
事件代理是代理模式最常見的一種應用方式,它的場景是一個父元素下有多個子元素
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" name="viewport">
<meta content="yes" name="apple-mobile-web-app-capable">
<meta content="black" name="apple-mobile-web-app-status-bar-style">
<meta content="telephone=no,email=no" name="format-detection">
<meta name="App-Config" content="fullscreen=yes,useHistoryState=yes,transition=yes">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title></title>
</head>
<body>
<p>圖片串列--事件代理</p>
<ul id="ul_wrapper">
<li>
1、<img id="img_1" src="" alt="">
</li>
<li>
2、<img id="img_2" src="" alt="">
</li>
<li>
3、<img id="img_3" src="" alt="">
</li>
<li>
4、<img id="img_4" src="" alt="">
</li>
<li>
5、<img id="img_5" src="" alt="">
</li>
<li>
6、<img id="img_6" src="" alt="">
</li>
<li>
7、<img id="img_7" src="" alt="">
</li>
<li>
8、<img id="img_8" src="" alt="">
</li>
</ul>
<script>
// 自己找個base64,拷貝上來太長了
let defualtSrc = https://www.cnblogs.com/longbensong/archive/2023/03/25/``
let initPage = (function () {
document.querySelectorAll('img').forEach(item => {
item.src = https://www.cnblogs.com/longbensong/archive/2023/03/25/defualtSrc
})
})();
document.querySelector('#ul_wrapper').addEventListener('click', function (e) {
if (e.target.nodeName === 'IMG') {
alert('圖片被點擊')
}
}, false)
</script>
</body>
</html>
快取代理
??快取代理可以避免重復的計算
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" name="viewport">
<meta content="yes" name="apple-mobile-web-app-capable">
<meta content="black" name="apple-mobile-web-app-status-bar-style">
<meta content="telephone=no,email=no" name="format-detection">
<meta name="App-Config" content="fullscreen=yes,useHistoryState=yes,transition=yes">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title></title>
</head>
<body>
<p>圖片串列--快取代理</p>
<button type="button">計算</button>
<div>
<label>結果:</label><input type="text">
</div>
<script>
/*
有效的減少計算;
工具函式
*/
const addAll = function (...args) {
console.log('進行了一次新計算')
let result = 0
const len = args.length
for (let i = 0; i < len; i++) {
result += args[i]
}
return result
}
let proxyAddAll = (function () {
const resultCache = {}
return function (fn, ...args) {
const key = args.join('')
if (resultCache[key]) {
return resultCache[key]
}
return resultCache[key] = fn.apply(this, args)
}
})()
// 123456 引數相同,只是第一次運算的時候,列印了一次進行了一次新計算
console.log(proxyAddAll(addAll, 1, 2, 3, 4, 5, 6))
console.log(proxyAddAll(addAll, 1, 2, 3, 4, 5, 6))
console.log(proxyAddAll(addAll, 1, 2, 3, 4, 5, 6))
// 1234567 因為是一個全新的引數所以列印了一次進行了一次新計算
console.log(proxyAddAll(addAll, 1, 2, 3, 4, 5, 7))
</script>
</body>
</html>
虛擬代理
??圖片預加載,預加載主要是為了避免網路不好、或者圖片太大時,頁面長時間給用戶留白的尷尬,原理也很簡單創建一個圖片實體指向圖片真實地址,當完成加載時,把占位圖的地址替換成真實的地址,這個時候瀏覽器會直接從快取里面拿,
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta content="width=device-width, initial-scale=1.0, maximum-scale=1.0, user-scalable=no" name="viewport">
<meta content="yes" name="apple-mobile-web-app-capable">
<meta content="black" name="apple-mobile-web-app-status-bar-style">
<meta content="telephone=no,email=no" name="format-detection">
<meta name="App-Config" content="fullscreen=yes,useHistoryState=yes,transition=yes">
<meta name="viewport" content="width=device-width, initial-scale=1">
<title></title>
</head>
<body>
<p>圖片串列--虛擬代理</p>
<ul>
<li>
1、<img id="img_1" src="" alt="">
</li>
<li>
2、<img id="img_2" src="" alt="">
</li>
<li>
3、<img id="img_3" src="" alt="">
</li>
<li>
4、<img id="img_4" src="" alt="">
</li>
<li>
5、<img id="img_5" src="" alt="">
</li>
<li>
6、<img id="img_6" src="" alt="">
</li>
<li>
7、<img id="img_7" src="" alt="">
</li>
<li>
8、<img id="img_8" src="" alt="">
</li>
</ul>
<script>
// 替換成你的base64,拷貝上來太長
let defualtSrchttps://www.cnblogs.com/longbensong/archive/2023/03/25/= ""
let initPage = (function () {
document.querySelectorAll('img').forEach(item => {
item.src = https://www.cnblogs.com/longbensong/archive/2023/03/25/defualtSrc
})
})();
// 設定圖片地址
function setImgUrl(dom, src) {
dom.src = src;
}
// 中間的代理圖片地址
function proxyImg(element, url) {
// 創建一個虛擬Image實體
const virtualImage = new Image()
virtualImage.onload = function () {
setImgUrl(element, url)
}
virtualImage.src = url
}
function preLoadImg() {
const urlList = ['https://t7.baidu.com/it/u=2621658848,3952322712&fm=193&f=GIF',
'https://t7.baidu.com/it/u=4080826490,615918710&fm=193&f=GIF',
'https://t7.baidu.com/it/u=334080491,3307726294&fm=193&f=GIF',
'https://t7.baidu.com/it/u=3713375227,571533122&fm=193&f=GIF',
'https://t7.baidu.com/it/u=801209673,1770377204&fm=193&f=GIF',
'https://t7.baidu.com/it/u=1856946436,1599379154&fm=193&f=GIF',
'https://t7.baidu.com/it/u=1010739515,2488150950&fm=193&f=GIF',
'https://t7.baidu.com/it/u=813347183,2158335217&fm=193&f=GIF']
document.querySelectorAll('img').forEach((element, index) => {
proxyImg(element, urlList[index])
element.src = https://www.cnblogs.com/longbensong/archive/2023/03/25/defualtSrc
})
}
setTimeout(() => {
preLoadImg()
}, 0.5 * 1000);
/*
核心: 有個虛擬的實體去請求地址,拿到之后替換到真實的dom
*/
</script>
