代理模式 (Proxy Pattern)

文章目录

      • 代理模式 (Proxy Pattern)
      • 原理
      • 分类
      • 优点
      • 缺点
      • 示例代码
        • 静态代理
          • 1. 定义接口
          • 2. 创建真实类
          • 3. 创建代理类
          • 4. 客户端代码
          • 输出结果
        • 动态代理(基于 JDK)
          • 1. 定义接口和真实类
          • 2. 创建动态代理类
          • 3. 客户端代码
          • 输出结果
      • UML 类图
        • 静态代理
        • 动态代理
      • 使用场景
      • 小结

代理模式 (Proxy Pattern)

代理模式是一种 结构型设计模式,它为某个对象提供一个替代或占位符,用来控制对这个对象的访问。通过代理对象,可以在实际对象的前后加入额外的功能(如权限校验、日志记录等)。


原理

  1. 核心思想
    • 使用一个代理类代表实际对象,客户端通过代理类间接访问实际对象。
  2. 适用场景
    • 控制对实际对象的访问(如权限验证)。
    • 增强功能(如日志记录、延迟加载)。
    • 远程代理、虚拟代理等场景。
  3. 参与角色
    • Subject(抽象主题类):定义真实对象和代理对象的公共接口。
    • RealSubject(真实主题类):实现抽象主题,表示被代理的真实对象。
    • Proxy(代理类):实现抽象主题,内部持有真实主题的引用,用来控制访问和增强功能。

分类

  1. 静态代理:代理类在编译时生成,开发者需要手动编写代理类。
  2. 动态代理:代理类在运行时动态生成,无需手动编写。
    • 基于 JDK 的动态代理(必须实现接口)。
    • 基于 CGLIB 的动态代理(无需接口)。

优点

  1. 控制访问:可以对访问真实对象进行约束。
  2. 功能扩展:通过代理类,在真实对象前后加入增强功能。

缺点

  1. 增加复杂性:引入代理类后,代码复杂度增加。
  2. 性能开销:动态代理会带来一定的性能消耗。

示例代码

静态代理

场景:一个 Payment 接口,有具体的支付实现。通过代理类对支付操作进行日志记录。


1. 定义接口
// 抽象主题
public interface Payment {void pay(int amount);
}

2. 创建真实类
// 真实主题类
public class RealPayment implements Payment {@Overridepublic void pay(int amount) {System.out.println("Processing payment of $" + amount);}
}

3. 创建代理类
// 代理类
public class PaymentProxy implements Payment {private RealPayment realPayment;public PaymentProxy(RealPayment realPayment) {this.realPayment = realPayment;}@Overridepublic void pay(int amount) {System.out.println("Logging: Initiating payment...");realPayment.pay(amount); // 调用真实对象的方法System.out.println("Logging: Payment completed.");}
}

4. 客户端代码
public class StaticProxyExample {public static void main(String[] args) {RealPayment realPayment = new RealPayment(); // 创建真实对象Payment proxy = new PaymentProxy(realPayment); // 创建代理对象proxy.pay(100); // 使用代理对象}
}

输出结果
Logging: Initiating payment...
Processing payment of $100
Logging: Payment completed.

动态代理(基于 JDK)

场景:通过动态代理实现相同的功能,无需手动编写代理类。


1. 定义接口和真实类
// 抽象主题
public interface Payment {void pay(int amount);
}// 真实主题类
public class RealPayment implements Payment {@Overridepublic void pay(int amount) {System.out.println("Processing payment of $" + amount);}
}

2. 创建动态代理类
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;// 动态代理类
public class DynamicProxyHandler implements InvocationHandler {private Object realObject;public DynamicProxyHandler(Object realObject) {this.realObject = realObject;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("Logging: Before method call...");Object result = method.invoke(realObject, args); // 调用真实对象的方法System.out.println("Logging: After method call...");return result;}// 获取代理实例public static Object getProxyInstance(Object realObject) {return Proxy.newProxyInstance(realObject.getClass().getClassLoader(),realObject.getClass().getInterfaces(),new DynamicProxyHandler(realObject));}
}

3. 客户端代码
public class DynamicProxyExample {public static void main(String[] args) {RealPayment realPayment = new RealPayment(); // 创建真实对象Payment proxy = (Payment) DynamicProxyHandler.getProxyInstance(realPayment); // 获取代理实例proxy.pay(200); // 使用代理对象}
}

输出结果
Logging: Before method call...
Processing payment of $200
Logging: After method call...

UML 类图

静态代理
    +------------+       +-------------+|  Payment   |<------+ RealPayment |+------------+       +-------------+^                     ^|                     |+-------------------------------+|         PaymentProxy          |+-------------------------------+
动态代理
    +------------+       +-------------+|  Payment   |<------+ RealPayment |+------------+       +-------------+^|+-------------+| Proxy       |+-------------+

使用场景

  1. 远程代理:为远程对象提供本地代理(如 RMI)。
  2. 虚拟代理:延迟加载资源,按需加载实际对象。
  3. 保护代理:对访问进行权限控制。
  4. 日志、性能监控:在方法调用前后增加日志或监控逻辑。

小结

  • 静态代理:需要手动编写代理类,适用于代理逻辑较少且稳定的场景。
  • 动态代理:无需手动编写代理类,灵活性强,适用于接口较多或逻辑频繁变化的场景。
  • 代理模式解耦了客户端与实际对象,增强了系统的灵活性和扩展性。

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

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

相关文章

【c语言】文件操作详解 - 从打开到关闭

文章目录 1. 为什么使用文件&#xff1f;2. 什么是文件&#xff1f;3. 如何标识文件&#xff1f;4. 二进制文件和文本文件&#xff1f;5. 文件的打开和关闭5.1 流和标准流5.1.1 流5.1.2 标准流 5.2 文件指针5.3 文件的打开和关闭 6. 文件的读写顺序6.1 顺序读写函数6.2 对比一组…

视频质量评价SimpleVQA

目录 一、研究意义例子 二、介绍三、文章解读3.1 论文动机3.2论文思路3.3方法3.3.1网络框架3.3.2公式解读3.3.3核心创新3.3.4理解 &#xff01;&#xff01;&#xff01;作者对模型的改进本人算法框体视频抽帧美学特征提取网络&#xff1a;3.3.5实验细节&#xff1a; 四、代码复…

【真实场景面试问题-2】

1 介绍一下低功耗设计手段–clock gating 和 clock domain&#xff1b; 芯片的低功耗设计尤其是在关注能耗的场景&#xff1a;移动设备、物联网和嵌入式系统&#xff1b;时钟门控&#xff08;Clock Gating&#xff09;和时钟域&#xff08;Clock Domain&#xff09;是两种常用…

c++编程玩转物联网:使用芯片控制8个LED实现流水灯技术分享

在嵌入式系统中&#xff0c;有限的GPIO引脚往往限制了硬件扩展能力。74HC595N芯片是一种常用的移位寄存器&#xff0c;通过串行输入和并行输出扩展GPIO数量。本项目利用树莓派Pico开发板与74HC595N芯片&#xff0c;驱动8个LED实现流水灯效果。本文详细解析项目硬件连接、代码实…

《白帽子讲Web安全》13-14章

《白帽子讲Web安全》13-14章 《白帽子讲Web安全》13-14章13、应用层拒绝服务攻击13.1、DDOS简介13.2、应用层DDOS13.2.1、CC攻击13.2.2、限制请求频率13.2.3、道高一尺&#xff0c;魔高一丈 13.3、验证码的那些事儿13.4、防御应用层DDOS13.5、资源耗尽攻击13.5.1、Slowloris攻击…

Linux—进程概念学习-03

目录 Linux—进程学习—31.进程优先级1.1Linux中的进程优先级1.2修改进程优先级—top 2.进程的其他概念3.进程切换4.环境变量4.0环境变量的理解4.1环境变量的基本概念4.2添加环境变量—export4.3Linux中环境变量的由来4.4常见环境变量4.5和环境变量相关的命令4.6通过系统调用获…

pikachu文件上传漏洞通关详解

声明&#xff1a;文章只是起演示作用&#xff0c;所有涉及的网站和内容&#xff0c;仅供大家学习交流&#xff0c;如有任何违法行为&#xff0c;均和本人无关&#xff0c;切勿触碰法律底线 目录 概念&#xff1a;什么是文件上传漏洞一、客户端check二、MIME type三、getimagesi…

从零开始学GeoServer源码(二)添加支持arcgis切片功能

文章目录 参考文章环境背景1、配置打包好的程序1.1、下载GeoServer的war包1.2、下载GeoWebCache1.3、拷贝jar包1.4、修改配置文件1.4.1、拷贝geowebcache-arcgiscache-context.xml1.4.2、修改geowebcache-core-context.xml1.4.3、修改geowebcache-servlet.xml 1.5、配置切片信息…

学习日志016--python实现双向循环列表与链栈

python中一些复合数据结构通过类的封装来实现的。双向循环链表与链栈也在其中。 双向循环链表 双向循环链表是一种特殊类型的链表&#xff0c;它结合了双向链表和循环链表的特点。在双向循环链表中&#xff0c;每个节点不仅包含数据&#xff0c;还持有指向前一个和后一个节点的…

【自动化Selenium】Python 网页自动化测试脚本(上)

目录 1、Selenium介绍 2、Selenium环境安装 3、创建浏览器、设置、打开 4、打开网页、关闭网页、浏览器 5、浏览器最大化、最小化 6、浏览器的打开位置、尺寸 7、浏览器截图、网页刷新 8、元素定位 9、元素交互操作 10、元素定位 &#xff08;1&#xff09;ID定位 &…

微软Ignite 2024:建立一个Agentic世界!

在今年的Microsoft Ignite 2024上&#xff0c;AI Agent无疑成为本次大会的重点&#xff0c;已经有十万家企业通过Copilot Studio创建智能体了。微软更是宣布&#xff1a;企业可以在智能体中&#xff0c;使用Azure目录中1800个LLM中的任何一个模型了&#xff01; 建立一个Agent…

嵌入式linux系统中图像处理基本方法

目录 2.1 BMP图像处理 2.1.1 BMP文件格式解析 2.1.2 代码实现:将BMP文件解析为RGB格式,在LCD上显示 2.2 JPEG图像处理 2.2.1 JPEG文件格式和libjpeg编译 2.2.2 libjpeg接口函数的解析和使用 2.2.3 使用libjpeg把JPEG文件解析为RGB格式,在LCD上显示 …

探索 GAN 的演变之路

2014 年&#xff0c;在论文Generative Adversarial Networks中&#xff0c;首次提出了 GAN&#xff0c;其核心思想是“生成”与“对抗”。GAN 由一个生成器 G(Generator)和一个判别器 D(Discriminator)构成&#xff0c;前者用于捕捉数据分布&#xff0c;后者用于判别某个样本是…

Vue实训---5-路由搭建

回顾之前的代码 我们在my-vue-project\src\router\index.js中的代码如下&#xff1a; // 什么是路由&#xff1f;路由就是url地址和组件的对应关系 // 1.引入vue-router import { createRouter, createWebHashHistory } from vue-router// 2.定义路由 const routes [{path: …

【GAMES101笔记速查——Lecture 19 Cameras,Lenses and Light Fields】

本章节内容&#xff1a;相机、棱镜、光场 计算机图形学的两种成像方法&#xff1a; 1.合成方法&#xff1a;光栅化、光线追踪&#xff08;展示出现实没有的东西&#xff09; 2.捕捉方法&#xff1a;相机&#xff08;捕捉现实已有的东西&#xff09; 目录 1 相机 1.1 针孔相…

MacOS系统上Jmeter 录制脚本遇到的证书坑位

一、JMeter介绍与安装 1&#xff0c;下载及安装 jmeter官网地址 二、录制百度链接https请求时&#xff0c;需要导入jmeter相关证书到macos系统的更目录中. 导入方式&#xff0c;直接拖入mac的系统中&#xff0c;始终新人就可以&#xff1b; 三、jmeter 创建相关的录制组件…

软件团队的共担责任

问责制被认为是个人与其社会系统之间的纽带&#xff0c;它创造了一种将个人与其行为和绩效联系起来的身份关系。在入门系列的第一篇文章《超越工具和流程&#xff1a;成功软件开发团队的策略》中&#xff0c;我们介绍了问责制的概念&#xff0c;并提出了以下定义&#xff1a; …

【Python爬虫实战】深入解析 Scrapy:从阻塞与非阻塞到高效爬取的实战指南

&#x1f308;个人主页&#xff1a;易辰君-CSDN博客 &#x1f525; 系列专栏&#xff1a;https://blog.csdn.net/2401_86688088/category_12797772.html ​ 目录 前言 一、阻塞和非阻塞 &#xff08;一&#xff09;阻塞 &#xff08;二&#xff09;非阻塞 二、Scrapy的工作…

【Python数据分析五十个小案例】电影评分分析:使用Pandas分析电影评分数据,探索评分的分布、热门电影、用户偏好

博客主页&#xff1a;小馒头学python 本文专栏: Python数据分析五十个小案例 专栏简介&#xff1a;分享五十个Python数据分析小案例 在现代电影行业中&#xff0c;数据分析已经成为提升用户体验和电影推荐的关键工具。通过分析电影评分数据&#xff0c;我们可以揭示出用户的…

第八篇:CamX RawHdr Feature Enable

CamX RawHdr Feature Enable RawHdr feature介绍: 试用于拍照场景,输入3张Raw,输出一张Raw。 对应的pipeline: camxSWMFMergeRaw.xml (usecases: UsecaseZSL) featureGraph: RTRawHDRBayer2YUVJPEG ​ RT -> RawHdr -> Bayer2Yuv -> JPEG RTRawHDRBayer2YUVJPE…