【设计模式】代理模式

简介

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

代理模式的解决方案是:

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

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

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

适用场景

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

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

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

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

优点

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

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

缺点

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

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

类图

代码

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("张三");
}
}
相关推荐
shuaixio6 小时前
【C++代码整洁之道】第九章 设计模式和习惯用法
c++·设计模式·设计原则·常见设计模式·习惯用法
南宫生7 小时前
Java迭代器【设计模式之迭代器模式】
java·学习·设计模式·kotlin·迭代器模式
程序员小赵同学7 小时前
AI Agent设计模式二:Parallelization
开发语言·python·设计模式
千千寰宇7 小时前
[设计模式/Java] 设计模式之工厂方法模式【11】
设计模式
此木|西贝9 小时前
【设计模式】模板方法模式
java·设计模式·模板方法模式
coderzpw9 小时前
告别通勤选择困难症——策略模式
设计模式·策略模式
编程侦探10 小时前
【设计模式】原型模式:用“克隆”术让对象创建更灵活
c++·设计模式·原型模式
Niuguangshuo13 小时前
Python 设计模式:外观模式
python·设计模式·外观模式
诺亚凹凸曼14 小时前
23种设计模式-行为型模式-命令
设计模式