引言
代理模式(Proxy Design Pattern)在不改变原始类(或叫被代理类)代码的情况下,通过引入代理类来给原始类附加功能。
代理模式的关键角色包括:
- 抽象主题(Subject):定义了目标对象和代理对象的共同接口,这样一来在任何可以使用目标对象的地方都可以使用代理对象。
- 目标对象(Real Subject):也称为被代理对象,是具体业务逻辑的实际执行者。
- 代理对象(Proxy):负责代理目标对象,它持有对目标对象的引用,并在其自身的方法中调用目标对象的方法,同时还可以在调用前后进行一些其他的操作。
应用场景
代理模式可以应用于许多场景,以下是几个常见的应用场景:
- 远程代理(Remote Proxy):代理模式可以用来在客户端和远程对象之间建立代理对象,隐藏了实际的网络通信细节。客户端通过代理对象调用远程对象的方法,而无需关心网络通信的具体实现。
- 虚拟代理(Virtual Proxy):代理模式可以用来延迟加载资源密集或耗时的对象,只有当真正需要使用这些对象时,才会创建并加载真实的对象。虚拟代理可以在一定程度上提升系统性能和响应速度。
- 安全代理(Protection Proxy):代理模式可以用来控制对敏感对象的访问权限。代理对象可以在调用目标对象方法之前检查权限,如果没有足够的权限,则不允许访问。
- 缓存代理(Caching Proxy):代理模式可以用来缓存目标对象的计算结果,当相同的请求再次到达时,可以直接返回缓存的结果,避免重复计算,提高系统性能。
- 日志记录代理(Logging Proxy):代理模式可以在目标对象的方法执行前后进行日志记录,用于跟踪和调试系统运行过程中的操作。
- AOP(面向切面编程):代理模式是实现AOP的基础,可以通过代理对象在目标对象的方法执行前后插入切面逻辑,例如日志、事务管理等。
需要注意的是,代理模式并非适用于所有情况。在某些场景下,代理模式可能引入额外的复杂性和性能开销,需要根据具体问题和需求来决定是否使用代理模式。
编程示例
代理模式的实现方式有多种,常见的有静态代理和动态代理两种形式:
- 静态代理:在编译时期就已经确定代理关系,代理类和目标类的关系在代码中明确指定。
// 抽象主题
public interface Subject {
void request();
}
// 目标对象
@Slf4j
public class RealSubject implements Subject {
@Override
public void request() {
// 具体业务逻辑
LOGGER.info("开始处理请求");
}
}
@Slf4j
public class Proxy implements Subject {
private Subject realSubject;
public Proxy(Subject realSubject) {
this.realSubject = realSubject;
}
@Override
public void request() {
LOGGER.info("前置处理逻辑");
// 执行一些额外的操作
realSubject.request();
// 执行一些额外的操作
LOGGER.info("后置处理逻辑");
}
}
// 客户端
public class Client {
public static void main(String[] args) {
Subject realSubject = new RealSubject(); // 创建目标对象
Subject proxy = new Proxy(realSubject); // 创建代理对象
proxy.request(); // 通过代理对象调用目标对象的方法
}
}
- 动态代理:在运行时动态生成代理类,无需提前编写代理类。Java 中的动态代理主要通过
java.lang.reflect.Proxy
类和java.lang.reflect.InvocationHandler
接口实现。
// 抽象主题
public interface Subject {
void request();
}
// 目标对象
@Slf4j
public class RealSubject implements Subject {
@Override
public void request() {
// 具体业务逻辑
LOGGER.info("开始处理请求");
}
}
// InvocationHandler 实现类
public class DynamicProxy implements InvocationHandler {
private Object target;
public DynamicProxy(Object target) {
this.target = target;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args) throws Throwable {
// 执行一些额外的操作
LOGGER.info("前置处理逻辑");
Object result = method.invoke(target, args);
// 执行一些额外的操作
LOGGER.info("后置处理逻辑");
return result;
}
}
// 客户端
public class Client {
public static void main(String[] args) {
Subject realSubject = new RealSubject(); // 创建目标对象
InvocationHandler handler = new DynamicProxy(realSubject); // 创建 InvocationHandler 实现类
Subject proxy = (Subject) Proxy.newProxyInstance(
realSubject.getClass().getClassLoader(),
realSubject.getClass().getInterfaces(),
handler
); // 创建动态代理对象
proxy.request(); // 通过代理对象调用目标对象的方法
}
}
以上内容基于GPT创建和整理。
参考
关于作者
来自一线全栈程序员nine的八年探索与实践,持续迭代中。欢迎关注“雨林寻北”或添加个人卫星codetrend(备注技术)。
1.本站内容仅供参考,不作为任何法律依据。用户在使用本站内容时,应自行判断其真实性、准确性和完整性,并承担相应风险。
2.本站部分内容来源于互联网,仅用于交流学习研究知识,若侵犯了您的合法权益,请及时邮件或站内私信与本站联系,我们将尽快予以处理。
3.本文采用知识共享 署名4.0国际许可协议 [BY-NC-SA] 进行授权
4.根据《计算机软件保护条例》第十七条规定“为了学习和研究软件内含的设计思想和原理,通过安装、显示、传输或者存储软件等方式使用软件的,可以不经软件著作权人许可,不向其支付报酬。”您需知晓本站所有内容资源均来源于网络,仅供用户交流学习与研究使用,版权归属原版权方所有,版权争议与本站无关,用户本人下载后不能用作商业或非法用途,需在24个小时之内从您的电脑中彻底删除上述内容,否则后果均由用户承担责任;如果您访问和下载此文件,表示您同意只将此文件用于参考、学习而非其他用途,否则一切后果请您自行承担,如果您喜欢该程序,请支持正版软件,购买注册,得到更好的正版服务。
5.本站是非经营性个人站点,所有软件信息均来自网络,所有资源仅供学习参考研究目的,并不贩卖软件,不存在任何商业目的及用途
暂无评论内容