零基础设计模式——结构型模式 - 代理模式

第三部分:结构型模式 - 代理模式 (Proxy Pattern)

在学习了享元模式如何通过共享对象来优化资源使用后,我们来探讨结构型模式的最后一个模式——代理模式。代理模式为另一个对象提供一个替身或占位符以控制对这个对象的访问。

  • 核心思想:为其他对象提供一种代理以控制对这个对象的访问。

代理模式 (Proxy Pattern)

“为其他对象提供一种代理以控制对这个对象的访问。” (Provide a surrogate or placeholder for another object to control access to it.)

想象一下,你想看一张非常大的高清图片,但加载它需要很长时间。或者,你可能需要访问一个远程服务器上的资源,网络延迟很高。或者,你可能需要对某个操作进行权限检查,只有特定用户才能执行。

代理模式通过引入一个代理对象来间接访问真实对象(也称为主题对象或服务对象)。客户端与代理对象交互,代理对象再根据需要与真实对象交互。

  • 真实主题 (Real Subject):实际执行任务的对象,如大图片加载器、远程服务接口、需要权限的操作。
  • 代理 (Proxy):控制对真实主题访问的替身。它可以实现与真实主题相同的接口,使得客户端可以无缝切换。

1. 目的 (Intent)

代理模式的主要目的:

  1. 控制访问:代理可以控制客户端对真实对象的访问权限、时机或方式。
  2. 提供间接层:在客户端和真实对象之间引入一个间接层,可以在这个层面上执行额外的操作,如延迟加载、缓存、日志记录、权限验证等。
  3. 简化复杂性:代理可以隐藏真实对象的复杂性,例如远程调用的网络细节。

2. 生活中的例子 (Real-world Analogy)

  • 信用卡

    • 你的银行账户(真实主题)里有钱。
    • 信用卡(代理)是你访问银行账户资金的一种方式。当你刷卡时,信用卡公司会验证你的身份、检查账户余额(控制访问),然后才允许交易。
  • 经纪人/中介

    • 你想买卖股票(真实主题是股票交易所)。
    • 你通过股票经纪人(代理)进行操作。经纪人会处理交易的细节,你不需要直接与交易所打交道。
    • 房产中介(代理)帮助你买卖房屋(真实主题是房产本身和房主)。
  • 门禁系统

    • 大楼的某个区域(真实主题)是受限的。
    • 门禁卡或保安(代理)验证你的身份和权限,决定是否允许你进入。
  • 明星的经纪人

    • 明星(真实主题)很忙。
    • 经纪人(代理)处理明星的日程安排、商业洽谈等,过滤掉不必要的干扰,并代表明星处理事务。

3. 结构 (Structure)

代理模式通常包含以下角色:

  1. Subject (主题接口):定义了 RealSubject 和 Proxy 的共同接口。这样,在任何使用 RealSubject 的地方都可以使用 Proxy。
  2. RealSubject (真实主题):定义了 Proxy 所代表的真实实体。这是实际执行业务逻辑的对象。
  3. Proxy (代理类)
    • 保存一个引用使得代理可以访问实体。若 RealSubject 和 Subject 的接口相同,Proxy 会引用 Subject。
    • 提供一个与 Subject 的接口相同的接口,这样代理就可以用来替代实体。
    • 控制对实体的存取,并可能负责创建和删除它。
    • 其他功能依赖于代理的类型。
  4. Client (客户端):通过 Subject 接口与 RealSubject 或 Proxy 交互。
    在这里插入图片描述
    工作流程
  • 客户端请求操作时,它会调用 Proxy 对象的方法。
  • Proxy 对象可能会执行一些预处理操作(如权限检查、日志记录)。
  • 如果需要,Proxy 会创建或获取 RealSubject 对象的引用,并将请求委托给 RealSubject
  • RealSubject 执行实际的操作。
  • Proxy 可能会执行一些后处理操作(如结果缓存、日志记录),然后将结果返回给客户端。

4. 常见代理类型 (Types of Proxies)

根据代理的目的和实现方式,有几种常见的代理类型:

  1. 虚拟代理 (Virtual Proxy)

    • 目的:延迟加载昂贵的对象。当创建真实对象的开销很大时,虚拟代理会推迟真实对象的创建,直到客户端真正需要它为止。
    • 例子:显示一个包含大量图片的文档,图片对象(真实主题)可以在实际滚动到屏幕上时才由虚拟代理创建和加载。
  2. 远程代理 (Remote Proxy)

    • 目的:为位于不同地址空间(如另一台机器上)的对象提供本地代表。远程代理负责处理网络通信的细节(如序列化、连接管理),使得客户端感觉像在调用本地对象一样。
    • 例子:Java RMI (Remote Method Invocation) 中的 Stub 对象就是远程代理。
  3. 保护代理 (Protection Proxy)

    • 目的:控制对真实对象的访问权限。在调用真实对象的方法之前,保护代理会检查客户端是否具有相应的权限。
    • 例子:根据用户角色控制对某些敏感操作的访问。
  4. 智能引用代理 (Smart Reference / Smart Proxy)

    • 目的:在访问对象时执行一些额外的操作,如引用计数、加锁以控制并发访问、对象加载时记录日志等。
    • 例子:C++ 中的智能指针(如 std::shared_ptr)可以看作是一种智能引用代理,负责管理对象的生命周期。
  5. 缓存代理 (Caching Proxy)

    • 目的:为开销大的操作结果提供临时存储。当多个客户端请求相同的结果时,可以直接从缓存中返回,避免重复计算或请求。
    • 例子:Web 代理服务器缓存常用网页;应用程序缓存数据库查询结果。
  6. 日志代理 (Logging Proxy)

    • 目的:在方法调用前后记录日志信息。

5. 适用场景 (When to Use)

  • 当你需要延迟初始化一个开销很大的对象时(虚拟代理)。
  • 当你需要控制对一个对象的访问权限时(保护代理)。
  • 当你需要为一个远程对象提供本地代表时(远程代理)。
  • 当你需要在访问对象时执行一些附加操作,如日志记录、缓存、事务管理等(智能引用代理、缓存代理、日志代理)。
  • 当你希望为一个对象提供不同级别的访问权限时。

6. 优缺点 (Pros and Cons)

优点:

  1. 控制访问:代理模式的核心优势在于能够控制对真实对象的访问。
  2. 增强功能:可以在不修改真实对象代码的情况下,通过代理为其增加额外的功能(如延迟加载、权限控制、日志、缓存)。
  3. 降低耦合:客户端与真实对象解耦,客户端只与代理接口交互。
  4. 提高性能:通过虚拟代理延迟加载或缓存代理缓存结果,可以提高系统性能。
  5. 远程访问透明化:远程代理使得客户端可以像访问本地对象一样访问远程对象。

缺点:

  1. 增加系统复杂性:引入了额外的代理类,可能会增加系统的设计和实现的复杂度。
  2. 可能引入性能开销:由于请求需要通过代理转发,可能会增加一次间接调用,带来轻微的性能延迟。但通常这种开销被代理带来的好处(如延迟加载、缓存)所抵消或超过。
  3. 真实主题的接口依赖:代理类通常依赖于真实主题的接口,如果真实主题接口发生变化,代理类也可能需要修改。

7. 实现方式 (Implementations)

让我们以一个虚拟代理为例,实现一个图片加载器。真实图片对象加载开销大,我们希望在实际显示时才加载它。

主题接口 (Image - Subject)
// image.go (Subject interface)
package imaging// Image 主题接口
type Image interface {Display()GetFilename() string
}
// Image.java (Subject interface)
package com.example.imaging;// 主题接口
public interface Image {void display();String getFilename();
}
真实主题 (RealImage - RealSubject)
// real_image.go (RealSubject)
package imagingimport ("fmt""time"
)// RealImage 真实主题
type RealImage struct {filename string
}func NewRealImage(filename string) *RealImage {ri := &RealImage{filename: filename}ri.loadFromDisk() // 创建时即加载return ri
}func (ri *RealImage) GetFilename() string {return ri.filename
}func (ri *RealImage) loadFromDisk() {fmt.Printf("RealImage: Loading image '%s' from disk...\n", ri.filename)// 模拟耗时操作time.Sleep(2 * time.Second)fmt.Printf("RealImage: Image '%s' loaded.\n", ri.filename)
}func (ri *RealImage) Display() {fmt.Printf("RealImage: Displaying image '%s'\n", ri.filename)
}
// RealImage.java (RealSubject)
package com.example.imaging;// 真实主题
public class RealImage implements Image {private String filename;public RealImage(String filename) {this.filename = filename;loadFromDisk(); // 创建时即加载}@Overridepublic String getFilename() {return filename;}private void loadFromDisk() {System.out.printf("RealImage: Loading image '%s' from disk...%n", filename);try {// 模拟耗时操作Thread.sleep(2000);} catch (InterruptedException e) {Thread.currentThread().interrupt();System.err.println("Loading interrupted for " + filename);}System.out.printf("RealImage: Image '%s' loaded.%n", filename);}@Overridepublic void display() {System.out.printf("RealImage: Displaying image '%s'%n", filename);}
}
代理类 (ProxyImage - Proxy)
// proxy_image.go (Proxy)
package imagingimport "fmt"// ProxyImage 代理类 (虚拟代理)
type ProxyImage struct {filename  stringrealImage *RealImage // 指向真实对象的指针,延迟初始化
}func NewProxyImage(filename string) *ProxyImage {fmt.Printf("ProxyImage: Created proxy for image '%s'. Real image not loaded yet.\n", filename)return &ProxyImage{filename: filename, realImage: nil}
}func (pi *ProxyImage) GetFilename() string {return pi.filename
}func (pi *ProxyImage) Display() {if pi.realImage == nil { // 延迟加载fmt.Printf("ProxyImage: Real image '%s' needs to be displayed. Loading now...\n", pi.filename)pi.realImage = NewRealImage(pi.filename)}pi.realImage.Display() // 委托给真实对象
}
// ProxyImage.java (Proxy)
package com.example.imaging;// 代理类 (虚拟代理)
public class ProxyImage implements Image {private String filename;private RealImage realImage; // 指向真实对象的引用,延迟初始化public ProxyImage(String filename) {this.filename = filename;System.out.printf("ProxyImage: Created proxy for image '%s'. Real image not loaded yet.%n", filename);}@Overridepublic String getFilename() {return filename;}@Overridepublic void display() {if (realImage == null) { // 延迟加载System.out.printf("ProxyImage: Real image '%s' needs to be displayed. Loading now...%n", filename);realImage = new RealImage(filename);}realImage.display(); // 委托给真实对象}
}
客户端使用
// main.go (示例用法)
/*
package mainimport ("./imaging""fmt"
)func main() {fmt.Println("--- Client: Creating proxy images ---")image1 := imaging.NewProxyImage("photo1.jpg")image2 := imaging.NewProxyImage("document_scan.png")// 此时,真实图片尚未加载fmt.Printf("\nImage 1 Filename: %s\n", image1.GetFilename())fmt.Printf("Image 2 Filename: %s\n", image2.GetFilename())fmt.Println("\n--- Client: Requesting to display image1 ---")image1.Display() // 第一次调用 Display,会触发真实图片加载fmt.Println("\n--- Client: Requesting to display image1 again ---")image1.Display() // 第二次调用 Display,真实图片已加载,直接显示fmt.Println("\n--- Client: Requesting to display image2 ---")image2.Display() // 第一次调用 Display for image2,会触发加载
}
*/
// Main.java (示例用法)
/*
package com.example;import com.example.imaging.Image;
import com.example.imaging.ProxyImage;public class Main {public static void main(String[] args) {System.out.println("--- Client: Creating proxy images ---");Image image1 = new ProxyImage("photo1.jpg");Image image2 = new ProxyImage("document_scan.png");// 此时,真实图片尚未加载System.out.printf("%nImage 1 Filename: %s%n", image1.getFilename());System.out.printf("Image 2 Filename: %s%n", image2.getFilename());System.out.println("%n--- Client: Requesting to display image1 ---");image1.display(); // 第一次调用 display,会触发真实图片加载System.out.println("%n--- Client: Requesting to display image1 again ---");image1.display(); // 第二次调用 display,真实图片已加载,直接显示System.out.println("%n--- Client: Requesting to display image2 ---");image2.display(); // 第一次调用 display for image2,会触发加载}
}
*/

8. 与装饰器模式的区别

代理模式和装饰器模式在结构上非常相似(都包装了另一个对象并实现了相同的接口),但它们的意图截然不同:

  • 代理模式 (Proxy)

    • 意图:控制对对象的访问。代理决定客户端是否、何时以及如何访问真实对象。
    • 关注点:访问控制、生命周期管理(如虚拟代理)、通信(如远程代理)。
    • 客户端感知:客户端可能不知道它正在与代理交互(例如,远程代理或保护代理对客户端是透明的),也可能知道(例如,客户端显式创建一个虚拟代理)。
    • 谁创建:代理通常由系统或框架创建和管理,或者客户端在特定场景下创建(如虚拟代理)。
  • 装饰器模式 (Decorator)

    • 意图:动态地向对象添加额外的职责或行为,而不改变其接口。
    • 关注点:增强对象的功能。
    • 客户端感知:客户端通常知道它正在使用一个装饰过的对象,并且通常负责构建装饰链。
    • 谁创建:装饰器通常由客户端根据需要动态地组合和应用。

简单来说:

  • 代理:我是“替身”或“看门的”,我管着你怎么用那个真实的东西。
  • 装饰器:我是“加料的”,我给那个真实的东西增加新花样。

9. 总结

代理模式是一种强大的结构型模式,它通过引入一个代理对象来控制对真实对象的访问。这种间接性使得我们可以在不修改真实对象代码的情况下,实现诸如延迟加载、权限控制、远程访问、日志记录、缓存等多种功能。根据具体需求,可以选择不同类型的代理(虚拟代理、保护代理、远程代理等)来解决特定的问题。

记住它的核心:提供替身,控制访问,增强间接性

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

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

相关文章

【OSS】 前端如何直接上传到OSS 上返回https链接,如果做到OSS图片资源加密访问

使用阿里云OSS(对象存储服务)进行前端直接上传并返回HTTPS链接,同时实现图片资源的加密访问,可以通过以下步骤实现: 前端直接上传到OSS并返回HTTPS链接 设置OSS Bucket: 确保你的OSS Bucket已创建&#xf…

TDenigne 集群可视化管理

可视化管理工具 为方便用户更高效地使用和管理 TDengine,TDengine 3.0 版本推出了一个全新的可视化组件 taosExplorer。这个组件旨在帮助用户在不熟悉 SQL 的情况下,也能轻松管理 TDengine 集群。通过 taosExplorer,用户可以轻松查看 TDengi…

Centos7安装gitlab

环境准备: 操作系统:Centos7 内存:2G以上 磁盘:50G 安全:关闭防火墙,selinux 1、安装GitLab所需依赖 yum -y install policycoreutils openssh-server openssh-clients postfix 2、设置postfix开机自启…

【前端面经】云智慧一面

写在前面:面经只是记录博主遇到的题目。每题的答案在编写文档的时候已经有问过deepseek,它只是一种比较普世的答案,要学得深入还是靠自己 Q:手撕代码,两个有序数组排序 A: function mysort(arr1, arr2) {…

Leetcode 3568. Minimum Moves to Clean the Classroom

Leetcode 3568. Minimum Moves to Clean the Classroom 1. 解题思路2. 代码实现 题目链接:3568. Minimum Moves to Clean the Classroom 1. 解题思路 这一题我的核心思路就是广度优先遍历遍历剪枝。 显然,我们可以给出一个广度优先遍历来给出所有可能…

Spring Boot,注解,@RestController

RestController 是 Spring MVC 中用于创建 RESTful Web 服务的核心注解。 RestController 核心知识点 REST 作用: RestController 是一个方便的组合注解,它结合了 Controller 和 ResponseBody 两个注解。 Controller: 将类标记为一个控制器,使其能够处理…

【计算机网络】Linux下简单的UDP服务器(超详细)

套接字接口 我们把服务器封装成一个类,当我们定义出一个服务器对象后需要马上初始化服务器,而初始化服务器需要做的第一件事就是创建套接字。 🌎socket函数 这是Linux中创建套接字的系统调用,函数原型如下: int socket(int domain, int typ…

Fashion-MNIST LeNet训练

前面使用线性神经网络softmax 和 多层感知机进行图像分类,本次我们使用LeNet 卷积神经网络进行 训练,期望能捕捉到图像中的图像结构信息,提高识别精度: import torch import torchvision from torchvision import transforms f…

EasyRTC嵌入式音视频通信SDK助力1v1实时音视频通话全场景应用

一、方案概述​ 在数字化通信需求日益增长的今天,EasyRTC作为一款全平台互通的实时视频通话方案,实现了设备与平台间的跨端连接。它支持微信小程序、APP、PC客户端等多端协同,开发者通过该方案可快速搭建1v1实时音视频通信系统,适…

查看make命令执行后涉及的预编译宏定义的值

要查看 make 命令执行后涉及的预编译宏定义(如 -D 定义的宏)及其值,可以采用以下方法: 1. 查看 Makefile 中的宏定义 直接检查 Makefile 或相关构建脚本(如 configure、CMakeLists.txt),寻找 -…

【C/C++】面试常考题目

面试中最常考的数据结构与算法题,适合作为刷题的第一阶段重点。 ✅ 分类 & 推荐题目列表(精选 70 道核心题) 一、数组 & 字符串(共 15 题) 题目类型LeetCode编号两数之和哈希表#1盛最多水的容器双指针#11三数…

【芯片学习】555

一、引脚作用 二、原理图 三、等效原理图 1.比较器 同相输入端大于反相输入端,输出高电平,反之亦然 2.三极管 给它输入高电平就可以导通 3.模拟电路部分 4.数字电路部分 这部分的核心是RS触发器,R-reset代表0,set是置位代表1&am…

Linux《文件系统》

在之前的系统IO当中已经了解了“内存”级别的文件操作,了解了文件描述符、重定向、缓冲区等概念,在了解了这些的知识之后还封装出了我们自己的libc库。接下来在本篇当中将会将视角从内存转向磁盘,研究文件在内存当中是如何进行存储的&#xf…

Java-代码段-http接口调用自身服务中的其他http接口(mock)-并建立socket连接发送和接收报文实例

最新版本更新 https://code.jiangjiesheng.cn/article/367?fromcsdn 推荐 《高并发 & 微服务 & 性能调优实战案例100讲 源码下载》 1. controller入口 ApiOperation("模拟平台端现场机socket交互过程,需要Authorization")PostMapping(path "/testS…

基于递归思想的系统架构图自动化生成实践

文章目录 一、核心思想解析二、关键技术实现1. 动态布局算法2. 样式规范集成3. MCP服务封装三、典型应用场景四、最佳实践建议五、扩展方向一、核心思想解析 本系统通过递归算法实现了Markdown层级结构到PPTX架构图的自动转换,其核心设计思想包含两个维度: 数据结构递归:将…

Python包管理器 uv替代conda?

有人问:python的包管理器uv可以替代conda吗? 搞数据和算法的把conda当宝贝,其他的场景能替代。 Python的包管理器有很多,pip是原配,uv是后起之秀,conda则主打数据科学。 uv替代pip似乎只是时间问题了,它…

使用pnpm、vite搭建Phaserjs的开发环境

首先,确保你已经安装了 Node.js 和 npm。然后按照以下步骤操作: 一、使用pnpm初始化一个新的 Vite 项目 pnpm create vite 输入名字 选择模板,这里我选择Vanilla,也可以选择其他的比如vue 选择语言 项目新建完成 二、安装相关依赖 进入项…

JS逆向案例—喜马拉雅xm-sign详情页爬取

JS逆向案例——喜马拉雅xm-sign详情页爬取 声明网站流程分析总结 声明 本文章中所有内容仅供学习交流,抓包内容、敏感网址、数据接口均已做脱敏处理,严禁用于商业用途和非法用途,否则由此产生的一切后果均与作者无关,若有侵权&am…

姜老师的MBTI课程:MBTI是可以转变的

我们先来看内向和外向这条轴,I和E内向和外向受先天遗传因素的影响还是比较大的,因为它事关到了你的硬件,也就是大脑的模型。但是我们在大五人格的排雷避坑和这套课程里面都强调了一个观点,内向和外向各有优势,也各有不…

进程同步:生产者-消费者 题目

正确答案: 问题类型: 经典生产者 - 消费者问题 同时涉及同步和互斥。 同步:生产者与消费者通过信号量协调生产 / 消费节奏(如缓冲区满时生产者等待,空时消费者等待)。互斥:对共享缓冲区的访问需…