【设计模式】代理模式

简介

假设你在网上购物时,快递员无法直接将包裹送到你手中(比如你不在家)。

代理模式的解决方案是:

快递员将包裹交给小区代收点(代理),代收点代替你控制和管理包裹的访问。

代收点可以添加额外功能:比如验证你的身份、暂存包裹直到你有空取件。

你最终通过代收点间接拿到包裹,整个过程保护了你的隐私和便利性。

适用场景

远程代理:访问远程对象(如网络服务调用)。

虚拟代理:延迟创建开销大的对象(如图片懒加载)。

保护代理:控制权限(如接口访问权限验证)。

日志代理:记录方法调用信息(如监控系统)。

优点

隔离目标对象与客户端,增强安全性。

在不修改目标对象的前提下扩展功能。

缺点

增加代码复杂度(需额外代理类)。

可能降低系统响应速度(代理层处理逻辑)。

类图

代码

java 复制代码
// 抽象主题:用户网购行为
interface Shopping {
void receivePackage(String packageName);
}


// 真实主题:快递直接送货
class CourierDelivery implements Shopping {
@Override
public void receivePackage(String packageName) {
System.out.println("快递员已将包裹[" + packageName + "]送到用户手中!");
}
}


// 代理类:代收点暂存包裹
class ParcelLockerProxy implements Shopping {
private Shopping courier;
private String pendingPackage;


public ParcelLockerProxy() {
this.courier = new CourierDelivery();
}


@Override
public void receivePackage(String packageName) {
// 代理控制访问:暂存包裹,延迟通知用户
System.out.println("代收点暂存包裹[" + packageName + "],等待用户取件");
this.pendingPackage = packageName;
notifyUser();
}


private void notifyUser() {
// 模拟延迟通知
System.out.println("短信通知:您的包裹已到代收点,请及时领取!");
}
}


// 客户端
public class ProxyDemo {
public static void main(String[] args) {
// 直接访问真实对象
Shopping courier = new CourierDelivery();
courier.receivePackage("手机");


// 通过代理访问
Shopping proxy = new ParcelLockerProxy();
proxy.receivePackage("电脑");
}
}

实际场景

Spring 框架使用动态代理实现面向切面编程(AOP),例如事务管理,日志代理。

java 复制代码
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;


// 接口:用户服务
interface UserService {
void addUser(String name);
}


// 真实主题:用户服务实现
class UserServiceImpl implements UserService {
@Override
public void addUser(String name) {
System.out.println("添加用户: " + name);
}
}


// 动态代理:日志增强
class LoggingProxy implements InvocationHandler {
private Object target;


public LoggingProxy(Object target) {
this.target = target;
}


@Override
public 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) {
UserService realService = new UserServiceImpl();
UserService proxy = (UserService) Proxy.newProxyInstance(
realService.getClass().getClassLoader(),
realService.getClass().getInterfaces(),
new LoggingProxy(realService)
);


proxy.addUser("张三");
}
}
相关推荐
星空寻流年10 小时前
设计模式第五章(门面模式)
android·设计模式
大飞pkz12 小时前
【设计模式】策略模式
开发语言·设计模式·c#·策略模式
yujkss12 小时前
23种设计模式之【状态机模式】-核心原理与 Java实践
java·ui·设计模式
bkspiderx17 小时前
C++设计模式之行为型模式:职责链模式(Chain of Responsibility)
c++·设计模式·责任链模式
青草地溪水旁18 小时前
设计模式(C++)详解——迭代器模式(4)
c++·设计模式·迭代器模式
大飞pkz18 小时前
【设计模式】状态模式
开发语言·设计模式·c#·状态模式
笨手笨脚の19 小时前
设计模式-建造者模式
java·设计模式·建造者模式·创建型设计模式
charlie11451419119 小时前
精读C++20设计模式——创造型设计模式:单例模式
c++·学习·单例模式·设计模式·c++20
1710orange20 小时前
java设计模式:工厂方法 + 建造者模式
java·设计模式
某不知名網友21 小时前
Reactor 模式:高并发网络编程的事件驱动利器
网络·设计模式·php