当前位置: 首页 > news >正文

代理模式:控制对象访问的中间层设计

代理模式:控制对象访问的中间层设计

一、模式核心:通过代理对象控制对目标对象的访问

在软件开发中,有时需要为对象添加一个 “代理” 来控制对它的访问,例如:

  • 远程代理:访问远程对象时(如 RPC 调用),本地代理对象伪装成远程对象
  • 虚拟代理:延迟加载目标对象(如图片未加载时显示占位符)
  • 保护代理:控制对敏感对象的访问权限(如用户权限校验)

** 代理模式(Proxy Pattern)** 通过引入代理类,在不修改目标对象的前提下,为其添加额外的逻辑或控制访问,核心解决:

  • 访问控制:在目标对象访问前后添加预处理逻辑
  • 资源优化:延迟初始化昂贵的目标对象
  • 职责分离:将非业务逻辑(如日志、事务)从目标对象中剥离

核心思想与 UML 类图(PlantUML 语法)

代理模式包含目标接口(Subject)、目标对象(RealSubject)和代理对象(Proxy),代理对象与目标对象实现相同接口,客户端通过代理间接访问目标:

PlantUML Diagram

二、核心实现:虚拟代理实现图片延迟加载

1. 定义目标接口(图片)

public interface Image {void display(); // 显示图片
}

2. 实现真实主题(实际图片对象,加载耗时)

public class RealImage implements Image {private String fileName;public RealImage(String fileName) {this.fileName = fileName;loadFromDisk(fileName); // 模拟加载耗时操作}private void loadFromDisk(String fileName) {System.out.println("加载图片:" + fileName);try {Thread.sleep(1000); // 模拟IO延迟} catch (InterruptedException e) {e.printStackTrace();}}@Overridepublic void display() {System.out.println("显示图片:" + fileName);}
}

3. 实现代理对象(延迟加载,先显示占位符)

public class ImageProxy implements Image {private String fileName;private RealImage realImage; // 真实对象延迟初始化public ImageProxy(String fileName) {this.fileName = fileName;}@Overridepublic void display() {if (realImage == null) {System.out.println("显示占位符..."); // 前置处理realImage = new RealImage(fileName); // 首次调用时加载真实对象}realImage.display(); // 调用真实对象的显示方法System.out.println("记录访问日志:" + fileName); // 后置处理}
}

4. 客户端通过代理访问目标对象

public class ClientDemo {public static void main(String[] args) {// 使用代理对象,无需直接创建RealImageImage image = new ImageProxy("large_image.jpg");System.out.println("第一次显示图片:");image.display(); // 触发延迟加载System.out.println("\n第二次显示图片:");image.display(); // 直接使用缓存的RealImage}
}

输出结果

第一次显示图片:
显示占位符...
加载图片:large_image.jpg
显示图片:large_image.jpg
记录访问日志:large_image.jpg第二次显示图片:
显示图片:large_image.jpg
记录访问日志:large_image.jpg

三、进阶:实现分布式代理与动态代理

1. 远程代理(模拟 RPC 调用)

// 远程服务接口(目标接口)
public interface RemoteService {String process(String request);
}// 本地代理对象(伪装成远程服务)
public class RemoteServiceProxy implements RemoteService {private String serverUrl;public RemoteServiceProxy(String serverUrl) {this.serverUrl = serverUrl;}@Overridepublic String process(String request) {// 通过HTTP/Netty发送请求到远程服务器return sendRequestToServer(request);}private String sendRequestToServer(String request) {System.out.println("发送远程请求:" + request + " 到 " + serverUrl);// 实际实现需处理网络IO和序列化return "远程响应:" + request.toUpperCase();}
}

2. 动态代理(基于 Java 反射,无需手动编写代理类)

import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;// 动态代理处理器
public class DynamicProxyHandler implements InvocationHandler {private Object target; // 目标对象public DynamicProxyHandler(Object target) {this.target = target;}@Overridepublic Object invoke(Object proxy, Method method, Object[] args) throws Throwable {System.out.println("动态代理前置处理:" + method.getName());Object result = method.invoke(target, args); // 调用目标对象方法System.out.println("动态代理后置处理:" + method.getName());return result;}
}// 创建动态代理示例
public class DynamicProxyDemo {public static void main(String[] args) {RealImage realImage = new RealImage("small_image.jpg");// 生成动态代理对象Image proxy = (Image) Proxy.newProxyInstance(Image.class.getClassLoader(),new Class<?>[] { Image.class },new DynamicProxyHandler(realImage));proxy.display(); // 通过代理调用目标方法}
}

3. 可视化代理流程

客户端请求
代理对象接收请求
代理执行前置逻辑
调用真实对象
真实对象处理请求
代理执行后置逻辑
返回结果给客户端

四、框架与源码中的代理模式实践

1. Spring AOP(动态代理实现切面编程)

  • 通过JdkDynamicAopProxyCglibAopProxy生成代理对象,织入切面逻辑
// 代理对象调用目标方法时,自动执行@Before、@After等增强
@Service
public class UserService {@Transactionalpublic void updateUser(User user) {// 业务逻辑}
}

2. MyBatis 映射器(Mapper Proxy)

  • MapperProxy作为代理对象,将接口方法映射为 SQL 语句执行
public interface UserMapper {@Select("SELECT * FROM users WHERE id = #{id}")User getUser(int id);
}// MyBatis创建MapperProxy代理对象
UserMapper mapper = sqlSession.getMapper(UserMapper.class);
User user = mapper.getUser(1); // 代理对象处理SQL执行

3. JavaScript 代理(Proxy 对象)

  • ES6 提供Proxy用于创建代理对象,拦截目标对象的操作
const realObject = { name: "Alice" };
const proxy = new Proxy(realObject, {get(target, property) {console.log(`访问属性:${property}`);return target[property];}
});
console.log(proxy.name); // 输出:访问属性:name  Alice

五、避坑指南:正确使用代理模式的 3 个要点

1. 区分代理与装饰器模式

  • 代理模式:代理对象控制对目标对象的访问(目标对象可能不存在或需控制访问)
  • 装饰器模式:装饰对象增强目标对象的功能(目标对象必须存在且功能需扩展)

2. 处理代理对象的线程安全

  • 若代理对象持有目标对象的引用,需确保多线程环境下的线程安全
public class ThreadSafeProxy implements Subject {private RealSubject realSubject = new RealSubject();private final Object lock = new Object();@Overridepublic void request() {synchronized (lock) {realSubject.request();}}
}

3. 避免代理链过长

  • 多层代理可能导致性能损耗和调试困难,建议:
    • 合并同类代理(如日志代理与权限代理合并为一个复合代理)
    • 使用 AOP 替代多层手动代理

4. 反模式:滥用代理导致复杂度上升

  • 对于简单的功能增强,直接修改目标对象可能更高效,无需引入代理

六、总结:何时该用代理模式?

适用场景核心特征典型案例
远程对象访问目标对象位于远程服务器,需通过网络调用RPC 框架、微服务客户端
延迟加载与缓存目标对象创建昂贵,需延迟初始化图片 / 视频加载、大文件处理
权限控制与日志记录需要在目标对象访问前后添加通用逻辑权限系统、事务管理、性能监控
接口伪装与隔离需要为复杂系统提供简单接口或伪装实现细节模拟对象(测试场景)、遗留系统适配

代理模式通过 “中间层控制 + 解耦访问” 的设计,为对象访问添加了灵活的控制层,是构建可扩展、易维护系统的重要模式。下一篇我们将深入探讨策略模式,解析如何动态切换算法实现,敬请期待!

扩展思考:静态代理 vs 动态代理

类型实现方式灵活性适用场景
静态代理手动编写代理类(编译期确定)低(代理类与目标类强绑定)简单场景、无需频繁修改代理逻辑
动态代理通过反射或字节码生成(运行时创建)高(可动态适配多个目标类)框架级代理(如 AOP、ORM)

理解这种差异,能帮助我们在不同场景下选择更合适的代理实现方式。

相关文章:

  • 论文阅读 | 大模型工具调用控制的策略优化
  • Spark与Hadoop之间的联系与区别
  • 使用nodeJs的express+axios+cors做代理
  • 配置MambaIRv2: Attentive State Space Restoration的环境
  • Sql刷题日志(day5)
  • 说一下Redis的发布订阅模型和PipeLine
  • OpenBayes 一周速览|EasyControl 高效控制 DiT 架构,助力吉卜力风图像一键生成;TripoSG 单图秒变高保真 3D 模型
  • leetcode hot100尝试1
  • 零基础入门 Verilog VHDL:在线仿真与 FPGA 实战全流程指南
  • 鸿蒙中的并发线程间通信、线程间通信对象
  • 状态模式(State Pattern)详解
  • Python | 分层线性模型的实现及示例
  • 什么是鸿蒙南向开发?什么是北向开发?
  • PHP 反序列化原生类 TIPS字符串逃逸CVE 绕过漏洞属性类型特征
  • 集结号海螺捕鱼游戏源码解析(第二篇):水浒传捕鱼模块逻辑与服务器帧同步详解
  • 2025山东省职业院校技能大赛网络安全赛项样题
  • node.js 实战——(path模块 知识点学习)
  • 解决重装idea后破解jerbel的问题
  • (一)单机架构、应用数据分离架构、应用服务集群架构
  • JavaScript学习教程,从入门到精通,Ajax与Node.js Web服务器开发全面指南(24)
  • 解放日报:上海一季度GDP同比增长5.1%,两大新动能助推经济“开门红”
  • 对话地铁读书人|超市营业员朱先生:通勤时间自学心理学
  • 工程院院士应汉杰不再担任苏州大学校长
  • 复旦大学校友夫妇一次性捐赠10亿元,成立学敏高等研究院
  • 女子“伸腿阻拦高铁关门”被行拘,事件追踪:当时发生了什么?
  • 俄乌互指对方未遵守复活节临时停火提议