首页 > 编程语言 >JavaScript 设计模式之代理模式

JavaScript 设计模式之代理模式

时间:2022-10-21 10:01:06浏览次数:76  
标签:const 函数 JavaScript 代理 li str container 设计模式

JavaScript 设计模式之代理模式_代理模式

代理模式,代理(proxy)是一个对象,它可以用来控制对另一个对象的访问。

现在页面上有一个香港回归最想听的金典曲目列表:

<ul id="container">
<li>我的中国心</li>
<li>东方之珠</li>
<li>香港别来无恙</li>
<li>偏偏喜欢你</li>
<li>相亲相爱</li>
</ul>

需要给页面添加一个效果:每当用户点击列表中的项目时,都会弹出一条消息:​​我想听:${name}​​,大致思路是给每个​​li​​元素添加一个点击事件。如下所示:

<!DOCTYPE html>
<html lang="zh">

<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>代理模式</title>
</head>

<body>
<ul id="container">
<li>我的中国心</li>
<li>东方之珠</li>
<li>香港别来无恙</li>
<li>偏偏喜欢你</li>
<li>相亲相爱</li>
</ul>

<script>
const container = document.getElementById("container")

Array.prototype.forEach.call(container.children, node => {
node.addEventListener("click", function (e) {
e.preventDefault()
alert(`我想听: ${e.target.innerText}`)
})
})
</script>
</body>

</html>

这种方法可以满足要求,但这样做的缺点是性能开销,因为每个 ​​li​​ 标签都绑定到一个事件。如果列表中有数千个元素,是否绑定了数千个事件?

JavaScript 设计模式之代理模式_html_02

每个 ​​li​​​ 都有自己的事件处理机制,但不管是哪个 ​​li​​​,其实都是 ​​ul​​​ 的成员,这样可以将 ​​li​​​ 的事件委托给父级节点 ​​ul​​​,让 ​​ul​​​ 成为这些 ​​li​​ 的事件代理。

JavaScript 设计模式之代理模式_代理模式_03

这样,只需要为这些 ​​li​​ 元素绑定一个事件,即为父级元素绑定一个事件。

const container = document.getElementById('container')

container.addEventListener('click', function (e) {
if (e.target.nodeName === 'LI') {
e.preventDefault()
alert(`我想听: ${e.target.innerText}`)
}
})

这就是代理模式的一种使用场合,代理模式是本体不直接出现,而是让代理间接解决问题。

  • 在上面代理模式的代码中,​​li​​ 并没有直接处理点击事件,而是将其委托给父级元素​​ul​​。
  • 现实生活中,明星并不是直接出来谈生意,而是交给他们的经纪人,也就是明星的代理人。

代理模式的应用非常广泛,再来看另一个适用场景。假设有一个计算函数,参数是字符串,计算比较耗时。同时,这是一个纯函数,如果参数相同,则函数的返回值将相同。

function compute(str) {    
// 假设这个函数执行时间很长
console.info("===> 超级计算开始了……");
return `输入:${str}`;
}

现在需要给这个函数添加一个缓存函数:每次计算后,存储参数和对应的结果。在接下来的计算中,会先从缓存中查询计算结果。当然,可以直接修改这个函数的功能。但这并不好,因为缓存不是这个功能的固有特性。

说到缓存函数,在 《​​从源码中学习Javascript技巧​​》聊到其实现,其实现就是使用代理模式。

更好的解决方案是使用代理模式。

const cached = (fn) => {
const cache = Object.create(null);
return (str) => {
const hit = cache[str];
return hit || (cache[str] = fn(str));
};
};
const cacheCompute = cached(compute);
console.log(cacheCompute("DevPoint"));
console.log(cacheCompute("DevPoint"));
console.log(cacheCompute("juejin"));

这样,就可以在不修改原函数逻辑的情况下为其扩展计算函数,这是代理模式的另一种使用场景,它允许向原始对象本身添加额外的功能,而无需更改它。

标签:const,函数,JavaScript,代理,li,str,container,设计模式
From: https://blog.51cto.com/devpoint/5781208

相关文章

  • 设计模式—关于提高可扩展性(模块层面)的学习(更加从容的应对需求变更)
    上一节主要学习了方法的可扩展性以及怎么更好的扩展方法,本节主要学习模块的可扩展性以及怎么更好的扩展模块。我们可以把任何一个程序看成是模块+组织模块沟通,模块是组成......
  • 设计模式—关于如何提高代码复用(再也不用担心重复代码多啦)
    上一节主要学习创建型的三种设计模式是怎么使用的。如何利用创建型设计模式来指导我们更好的封装代码更好的创建对象,本节主要学习怎样利用设计模式来提高代码复用性。提高......
  • A* 自动寻路算法-JavaScript
    效果图代码<!DOCTYPEhtml><htmllang="en"><head><metacharset="UTF-8"><metahttp-equiv="X-UA-Compatible"content="IE=edge"><metaname="viewport"......
  • 初识设计模式 - 命令模式
    简介命令设计模式(CommandDesignPattern)可以将请求发送者和接收者完全解耦。发送者和接收者之间没有直接引用关系,发送请求的对象只需要知道如何发送请求,而不必知道如何完......
  • JavaScript实现数据结构 -- 栈
    栈栈是一种==后进先出==的数据结构。JS模拟栈虽然JavaScript中没有栈,但是我们可以用数组来实现栈的功能。 //定义一个数组用来模拟栈 conststack=[]; //用数组......
  • JavaScript实现数据结构 -- 队列
    队列队列是一个先进先出的数据结构。JS模拟队列虽然JavaScript中没有队列,但是我们可以用数组来实现队列的功能。 //用数组来模拟队列 constqueue=[]; //入队 q......
  • JavaScript实现数据结构 -- 链表
    链表链表和数组一样是有多个元素组成的列表;不同的是链表元素存储不连续,用next指针连接在一起;链表的特点插入、删除不需要移动元素;不必事先分配存储空间;所需空间与长......
  • JavaScript基础知识
    JavaScript基础知识##输出语句*1.window.alert()--写入警告框*2.document.write()---写入HTML输出*3.console.log()---写入浏览器控制台*alert("helloworld!......
  • JavaScript中的Promise
    阮一峰ES6入门Promise1.Promise的介绍Promise是异步编程的一种新的解决方案,从早期的回调函数、事件相比,更加合理和强大。语法上来说,Promise是一个对象,可以获取异......
  • 有关反向代理的记录
    反向代理服务器位于用户与目标服务器之间,但是对于用户而言,反向代理服务器就相当于目标服务器,即用户直接访问反向代理服务器就可以获得目标服务器的资源。同时,用户不需要知......