简介
说明
本文用示例介绍Java的代理模式的写法,包括:静态代理,动态代理。
代理模式介绍
为某个对象提供一个代理,以控制对这个对象的访问。 代理类和委托类有共同的父类或父接口,这样在任何使用委托类对象的地方都可以用代理对象替代。代理类负责请求的预处理、过滤、将请求分派给委托类处理、以及委托类执行完请求后的后续处理。
从图中可以看出,代理接口(Subject)、代理类(ProxySubject)、委托类(RealSubject)形成一个“品”字结构。
根据代理类的生成时间不同可以将代理分为静态代理和动态代理两种。
静态代理与动态代理区别
项 | 静态代理 | 动态代理 |
代理类字节码创建的时间 | 程序运行前就已存在代理类的字节码文件,代理类和委托类的关系在运行前就确定了。 程序员创建或工具生成代理类的源码,再编译代理类。 | 不存在代理类的字节码文件。动态代理类的源码是在程序运行期间由JVM根据反射等机制动态的生成的,代理类和委托类的关系是在程序运行时确定。 |
动态代理占优势的项 | ||
接口有多个方法的处理 | 如果接口增加一个方法,除了所有实现类需要实现这个方法外,所有代理类也需要实现此方法。 | 接口中声明的所有方法都被转移到调用处理器一个集中的方法中处理,接口方法数量比较多的时候,不需要像静态代理那样每一个方法进行中转。 在本示例中看不出来,因为invoke方法体内嵌入了具体的外围业务(记录任务处理前后时间并计算时间差),实际中可以类似Spring AOP那样配置外围业务。 |
JDK动态代理与CGLIB动态代理区别
项 | JDK动态代理 | CGLIB动态代理 |
接口是否需实现 | 只能代理实现了接口的类。 | 可以代理没有实现接口的类。 |
原理 | 继承自Proxy,而Proxy中有InvocationHandler的实现类的引用。 调用被代理的类的方法时,会调用父类(Proxy)的InvocationHandler的invoke方法。 | 对指定的目标类生成一个子类,并覆盖其中方法实现增强。 |
是否支持final class | 支持。 | 不支持。 因为CGLIB是生成子类来实现AOP。 |
执行效率 | 少量调用时:JDK代理效率高于CGLIB 大量调用时:jdk6比CGLIB效率低一点。JDK1.7与1.8效率高于CGLIB(1W次调用,高20%) | JDK1.6之前,效率比JDK高。 |
静态代理
时间统计示例
代码
package org.example.a;
//代理接口
interface ExecuteTime {
public void dealTask(String taskName);
}
//被代理类
class ExecuteTimeImpl implements ExecuteTime {
//这里打印出任务名,并休眠500ms模拟任务执行了很长时间
@Override
public void dealTask(String taskName) {
System.out.println("Task is running: "+taskName);
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
//代理类
class ExecuteTimeProxy implements ExecuteTime {
private ExecuteTime delegate = null;
public ExecuteTimeProxy(ExecuteTimeImpl ExecuteTimeImpl) {
this.delegate = ExecuteTimeImpl;
}
@Override
public void dealTask(String taskName) {
long startTime = System.currentTimeMillis();
delegate.dealTask(taskName);
long endTime = System.currentTimeMillis();
System.out.println("Elapsed time: "+(endTime - startTime)+" ms");
}
}
//工厂方法获得代理对象
class StatiProxyFactory {
//对客户类来说,其并不知道返回的是代理类对象还是委托类对象。
public static ExecuteTime getInstance(){
return new ExecuteTimeProxy(new ExecuteTimeImpl());
}
}
public class Demo{
public static void main(String[] args) {
ExecuteTime proxy = StatiProxyFactory.getInstance();
proxy.dealTask("TestTask");
}
}
运行结果:
Task is running: TestTask
Elapsed time: 501 ms