【前端设计模式】之代理模式

代理模式特性

代理模式是一种结构型设计模式,它通过创建一个代理对象来控制对另一个对象的访问。代理模式的主要特性包括:

  1. 代理对象与目标对象实现相同的接口或继承相同的基类,使得客户端可以透明地使用代理对象。
  2. 代理对象持有对目标对象的引用,并在必要时将客户端的请求转发给目标对象。
  3. 代理对象可以在转发请求之前或之后执行一些额外的操作,例如权限验证、缓存、日志记录等。

前端应用示例

请求代理

当发起请求时,可以代理请求在转发请求之前执行一些额外的操作,或者在转发请求之后做一些额外的操作

 
// 定义目标对象接口
class Subject {request() {// 处理请求}
}
// 定义具体目标对象类
class RealSubject extends Subject {request() {// 处理真实请求}
}
// 定义代理对象类
class Proxy extends Subject {constructor() {super();this.realSubject = new RealSubject();}request() {// 在转发请求之前或之后执行一些额外操作this.preRequest();this.realSubject.request();this.postRequest();}preRequest() {// 在转发请求之前执行一些额外操作}postRequest() {// 在转发请求之后执行一些额外操作}
}
// 使用示例
const proxy = new Proxy();
proxy.request(); // 通过代理对象发送请求,并在转发前后执行额外操作

图片懒加载

当页面中存在大量图片时,为了提高页面加载速度和性能,可以使用图片懒加载技术。在这种情况下,可以使用代理模式,在图片未进入可视区域之前,使用占位图或者小图进行替换,在图片进入可视区域时再加载真实图片。

// 定义目标对象
class ImageLoader {constructor(imageElement) {this.imageElement = imageElement;this.realImage = new Image();this.realImage.onload = () => {this.imageElement.src = this.realImage.src;};}load() {// 模拟从服务器加载真实图片的操作console.log("Loading real image...");this.realImage.src = this.imageElement.dataset.src;}
}
// 定义图片懒加载代理对象
class LazyImageProxy {constructor(imageElement) {this.imageElement = imageElement;}load() {if (this.imageElement.getBoundingClientRect().top < window.innerHeight) {const imageLoader = new ImageLoader(this.imageElement);imageLoader.load();}}
}
// 使用示例
const lazyImages = document.querySelectorAll(".lazy-image");
lazyImages.forEach((image) => {const lazyImageProxy = new LazyImageProxy(image);lazyImageProxy.load();
});

在上述示例中,我们定义了一个目标对象ImageLoader,它负责加载真实的图片。

然后,我们定义了一个图片懒加载代理对象LazyImageProxy,它持有对目标对象的引用,并在load方法中实现了对图片懒加载的逻辑。当客户端调用load方法时,代理对象首先检查图片是否进入可视区域(通过判断其位置是否小于窗口高度),如果是则创建目标对象ImageLoader并调用其load方法加载真实图片。

通过使用代理模式实现图片懒加载,可以减少页面加载时对大量图片的请求,提高页面加载速度和性能。同时,代理对象还可以隐藏目标对象的具体实现细节,保护目标对象的安全性。

数据缓存代理

为了减少网络请求和提高页面加载速度,可以使用代理模式在客户端或服务端缓存数据。当客户端请求数据时,代理对象首先检查缓存中是否存在数据,如果存在则直接返回缓存数据,否则再向目标对象请求数据并将其缓存起来。

 
// 定义目标对象
class DataService {fetchData(key) {// 模拟从服务器获取数据的操作console.log("Fetching data from server...");return `Data for ${key}`;}
}
// 定义数据缓存代理对象
class DataCacheProxy {constructor() {this.cache = {};this.dataService = new DataService();}fetchData(key) {if (this.cache[key]) {console.log("Fetching data from cache...");return this.cache[key];} else {const data = this.dataService.fetchData(key);this.cache[key] = data;return data;}}
}
// 使用示例
const proxy = new DataCacheProxy();
console.log(proxy.fetchData("example")); // 从服务器获取数据,并缓存起来
console.log(proxy.fetchData("example")); // 从缓存中获取数据

在上述示例中,我们定义了一个目标对象DataService,它模拟了从服务器获取数据的操作。然后,我们定义了一个数据缓存代理对象DataCacheProxy,它持有对目标对象的引用,并在fetchData方法中实现了对数据的缓存逻辑。

当客户端调用fetchData方法时,代理对象首先检查缓存中是否存在对应的数据,如果存在则直接返回缓存的数据;否则调用目标对象的fetchData方法从服务器获取数据,并将其缓存起来。

通过使用代理模式实现数据缓存代理,可以减少对服务器的请求次数,提高数据访问的性能和效率。同时,代理对象还可以隐藏目标对象的具体实现细节,保护目标对象的安全性。

ES6的Proxy

ES6引入了Proxy对象,它是一种代理模式的实现,用于拦截并自定义对象的操作。Proxy对象可以拦截并重定义JavaScript对象的底层操作,例如属性访问、赋值、函数调用等。通过使用Proxy对象,我们可以在目标对象上添加额外的行为或修改默认行为。

Proxy对象的基本语法如下:

const proxy = new Proxy(target, handler);
  • target:要代理的目标对象。
  • handler:一个包含各种拦截操作的处理程序对象。

下面是一些常见的Proxy拦截操作:

  1. get(target, property, receiver):拦截对目标对象属性的读取操作。
  2. set(target, property, value, receiver):拦截对目标对象属性的赋值操作。
  3. apply(target, thisArg, argumentsList):拦截对目标函数的调用操作。
  4. has(target, property):拦截in运算符判断属性是否存在于目标对象中。
  5. deleteProperty(target, property):拦截对目标对象属性的删除操作。

以下是一个简单示例,展示了如何使用ES6 Proxy来实现一个简单的权限控制:

const user = {name: "John",isAdmin: false,
};const userProxy = new Proxy(user, {get(target, property) {if (property === "isAdmin") {return false; // 拒绝访问isAdmin属性}return target[property];},set(target, property, value) {if (property === "isAdmin") {throw new Error("Cannot modify isAdmin property.");}target[property] = value;return true;},
});console.log(userProxy.name); // 输出: "John"
console.log(userProxy.isAdmin); // 输出: false
userProxy.isAdmin = true; // 抛出错误: "Cannot modify isAdmin property."

在上述示例中,我们创建了一个名为user的普通对象,并使用Proxy对象创建了一个名为userProxy的代理对象。在代理对象的处理程序中,我们拦截了对isAdmin属性的读取和赋值操作,并进行了相应的权限控制。

ES6 Proxy提供了强大的拦截能力,可以用于实现数据校验、权限控制、数据劫持等功能。然而,需要注意使用Proxy时要考虑性能问题,因为每个操作都会经过拦截处理。

优缺点

优点
  1. 代理模式可以实现对目标对象的访问控制,可以在不改变目标对象的情况下增加额外的功能。
  2. 通过使用代理模式,可以实现客户端与目标对象之间的解耦,提高代码的可维护性和可扩展性。
  3. 代理模式可以隐藏目标对象的具体实现细节,保护目标对象的安全性。
缺点
  1. 代理模式增加了系统的复杂性,引入了额外的代理对象。
  2. 在一些情况下,代理模式可能会导致请求的延迟,因为请求需要经过代理对象转发。

总结

代理模式是一种常用的设计模式,它通过创建一个代理对象来控制对另一个对象的访问。通过使用代理模式,可以实现对目标对象的访问控制、增加额外功能、解耦客户端与目标对象等。然而,需要根据具体情况权衡使用代理模式所带来的优缺点。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/87407.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

用flex实现grid布局

1. css代码 .flexColumn(columns, gutterSize) {display: flex;flex-flow: row wrap;margin: calc(gutterSize / -2);> div {flex: 0 0 calc(100% / columns);padding: calc(gutterSize / 2);box-sizing: border-box;} }2.用法 .grid-show-item3 {width: 100%;display: fl…

【SpringBoot】-IDEA社区版SpringBoot项目的创建

作者&#xff1a;学Java的冬瓜 博客主页&#xff1a;☀冬瓜的主页&#x1f319; 专栏&#xff1a;【Framework】 主要内容&#xff1a;IDEA下的springboot项目的创建&#xff0c;网页下springboot项目的创建。springboot目录结构的认识。 文章目录 一、什么是SpringBoot&#x…

【C++】STL之适配器---用deque实现栈和队列

目录 前言 一、deque 1、deque 的原理介绍 2、deque 的底层结构 3、deque 的迭代器 4、deque 的优缺点 4.1、优点 4.2、缺点 二、stack 的介绍和使用 1、stack 的介绍 2、stack 的使用 3、stack 的模拟实现 三、queue 的介绍和使用 1、queue 的介绍 2、queue 的使用 3、qu…

RabbitMQ实现秒杀场景示例

本文章通过MQ队列来实现秒杀场景 整体的设计如下图&#xff0c;整个流程中对于发送发MQ失败和发送到死信队列的数据未做后续处理 1、首先先创建MQ的配置文件 Configuration public class RabbitConfig {public static final String DEAD_LETTER_EXCHANGE "deadLetterE…

Python爬虫在Web应用自动化测试中的应用

在Web应用开发过程中&#xff0c;自动化测试是确保应用质量和稳定性的重要环节。本文将介绍如何使用Python爬虫与自动化测试技术相结合&#xff0c;实现对Web应用进行自动化测试的方法和步骤。通过这种结合&#xff0c;我们可以提高测试效率、减少人力成本&#xff0c;并确保应…

2023-09-25 LeetCode每日一题(LFU 缓存)

2023-09-25每日一题 一、题目编号 460. LFU 缓存二、题目链接 点击跳转到题目位置 三、题目描述 请你为 最不经常使用&#xff08;LFU&#xff09;缓存算法设计并实现数据结构。 实现 LFUCache 类&#xff1a; LFUCache(int capacity) - 用数据结构的容量 capacity 初始…

设计模式之解释器模式

文章目录 四则运算问题传统方案解决四则运算问题分析解释器模式基本介绍解释器模式的原理类图解释器模式来实现四则解释器模式的注意事项和细节 四则运算问题 通过解释器模式来实现四则运算&#xff0c;如计算 ab-c 的值&#xff0c;具体要求 先输入表达式的形式&#xff0c;比…

如何访问TDH中Inceptor 底层的元数据库TxSQL

如何访问TDH中Inceptor 底层的元数据库TxSQL 1 Inceptor概述 在大数据生态系统中&#xff0c;HIVE是离线数据仓库事实上的标准&#xff0c;绝大多数的大数据分析型系统或数据仓库系统&#xff0c;都是基于HIVE来构建的。 在星环的大数据平台TDH中&#xff0c;在功能上对应开…

基于物联网的农村地区智能微电网系统(Simulink)

&#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜密&#xff0c;逻辑清晰&#xff0c;为了方便读者。 ⛳️座右铭&a…

python 字符串

字符串回文关于字符大小写.capitalize().casefold().title().swapcase().upper().lower() 对齐函数中间对齐&#xff1a;.center(x&#xff0c;y)左对齐&#xff1a;.ljust(x,y)右对齐&#xff1a;.rjust(x,y)右对齐&#xff0c;左侧补零&#xff1a;zfill(x) 查找查找数量&…

uniapp webview实现双向通信

需求&#xff1a;uniapp webview嵌套一个h5 实现双向通信 uniapp 代码 <template><view><web-view src"http://192.168.3.150:9003/" message"onMessage"></web-view></view> </template><script>export defau…

全网首发YOLOv8暴力涨点:Gold-YOLO,遥遥领先,超越所有YOLO | 华为诺亚NeurIPS23

💡💡💡本文独家改进:提出了全新的信息聚集-分发(Gather-and-Distribute Mechanism)GD机制,Gold-YOLO,替换yolov8 head部分 实现暴力涨点 Gold-YOLO | 亲测在多个数据集能够实现大幅涨点 💡💡💡Yolov8魔术师,独家首发创新(原创),适用于Yolov5、Yolov7、…

Mysql主从数据恢复随笔

目录 1.使用pt-table-checksum插件安装方式如下 2.在主节点执行检查数据同步情况 3.同步检查出现的问题 3.1没有sock文件 3.2 Authentication plugin ‘sha256_password’ cannot be loaded: /usr/lib64/mysql/plugin/sha256_password.so: 无法打开共享对象文件: 没有那个文…

【办公自动化】使用Python一键往Word文档的表格中填写数据(文末送书)

&#x1f935;‍♂️ 个人主页&#xff1a;艾派森的个人主页 ✍&#x1f3fb;作者简介&#xff1a;Python学习者 &#x1f40b; 希望大家多多支持&#xff0c;我们一起进步&#xff01;&#x1f604; 如果文章对你有帮助的话&#xff0c; 欢迎评论 &#x1f4ac;点赞&#x1f4…

vSAN数据恢复-vSAN架构下虚拟机磁盘组件出现问题的数据恢复案例

vsan数据恢复环境&#xff1a; 一套VMware vSAN超融合基础架构&#xff0c;全闪存&#xff0c;开启压缩重删。共11台服务器节点。每台服务器节点上配置1块PCIE固态硬盘和8-10块SSD固态硬盘。 每个服务器节点上创建1个磁盘组&#xff0c;每个磁盘组将1个PCIE固态硬盘识别为2个硬…

YOLOv8快速复现 官网版本 ultralytics

YOLOV8环境安装教程.&#xff1a;https://www.bilibili.com/video/BV1dG4y1c7dH/ YOLOV8保姆级教学视频:https://www.bilibili.com/video/BV1qd4y1L7aX/ b站视频&#xff1a;https://www.bilibili.com/video/BV12p4y1c7UY/ 1 平台搭建YOLOv8 平台&#xff1a;https://www.a…

Halcon Tuple相关算子(一)

(1) tuple_length( : : Tuple : Length) 功能&#xff1a;返回输入元组中元素的个数。 控制输入参数&#xff1a; Tuple&#xff1a;输入元组&#xff1b; 控制输出参数&#xff1a;length&#xff1a;输入元组中元素的个数。 (2) tuple_find( : : Tuple, ToFind : Indices…

使用Python进行员工流失分析

员工流失分析是指分析离开公司、企业的员工的行为&#xff0c;并将他们与公司中的现有员工进行比较。它有助于找出哪些员工可能很快离开。所以&#xff0c;如果你想学习如何分析员工流失&#xff0c;这篇文章适合你。本文中&#xff0c;将带您完成使用Python进行员工流失分析的…

【李沐深度学习笔记】基础优化方法

课程地址和说明 基础优化方法p2 本系列文章是我学习李沐老师深度学习系列课程的学习笔记&#xff0c;可能会对李沐老师上课没讲到的进行补充。 基础优化方法 在讲具体的线性回归实现之前&#xff0c;要先讲一下基础的优化模型的方法 梯度下降 当模型没有显示解&#xff08…

华为孟晚舟:从最惨千金 到最强战士

作者&#xff1a;积溪 简评&#xff1a;华为25号开发布会&#xff0c;有何深意&#xff1f;从最惨千金到最强战士&#xff0c;孟晚舟和华为都回来了 #华为发布会 #孟晚舟 #任正非 #华为 华为发布会 在打谁的脸&#xff1f; 苹果只是前菜 今天才是正餐 两年前的今天 华为…