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

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

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

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

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

  • 远程代理:访问远程对象时(如 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)

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

http://www.xdnf.cn/news/1352.html

相关文章:

  • C#学习1_认识项目/程序结构
  • 【无标题】spark安装部署
  • TCP 协议:原理、机制与应用
  • cursor改Goland操作习惯
  • 密码学(1)LWE,RLWE,MLWE的区别和联系
  • 校园外卖服务系统的设计与实现(代码+数据库+LW)
  • Transformer起源-Attention Is All You Need
  • 考研系列-计算机组成原理第一章:计算机系统概述
  • 【论文精读】Reformer:高效Transformer如何突破长序列处理瓶颈?
  • 23种设计模式-结构型模式之组合模式(Java版本)
  • Netty的心跳机制怎么实现的?
  • uniapp返回上一页接口数据更新了,页面未更新
  • 嵌入式WebRTC音视频实时通话EasyRTC助力打造AIOT智能硬件实时通信新生态
  • 【小皮(PHPstudy】
  • Collection集合,List集合,set集合,Map集合
  • 主流单片机厂商/系列、型号、内核、主频、Flash、RAM、关键外设、特殊功能、典型应用及选型对比与分析,
  • 学习海康VisionMaster之卡尺工具
  • 【redis】主从复制
  • MATLAB 下载安装教程
  • ubuntu系统下部署使用git教程
  • 第五章:Benchmark Framework
  • C# .NET Core 批量下载文件
  • 【最新版】沃德代驾源码全开源+前端uniapp
  • rust编程学习(三):8大容器类型
  • Linux:42线程控制lesson30
  • [进程通信] 管道 vs 共享内存 vs 网络
  • Linus的权限问题
  • 强化学习和微调 区别如下
  • PostgreSQL认证培训推荐机构
  • 动态提示词(小模型)、RAG和提示词系统