在静态代理中对汽车创建了一个时间记录的代理,那么如果我们要对火车、飞机都记录时间呢?这就需要分别对火车、飞机创建代理类。我们需要一个方法来把记录时间这个代理抽离出来,这样当需要对不同的交通工具实现记录时间功能的时候,直接运用这个抽离出的代理,这样可大大减少代码的重用率。这就引出了动态代理。
JDK动态代理简述
动态代理就是动态产生代理,实现对不同类、不同方法的代理。以下为JDK动态代理的实现机制:
如图所示,JDK的动态代理其实就是在代理类ProxySubject与被代理类RealSubject之间加入了一个ProxyHandler类,这个ProxyHandler类实现了InvocationHandler接口,它充当事务处理器,比如类似于上面给汽车计时的日志处理,时间处理等事务都是在这个ProxyHandler类中完成的。
Java动态代理类位于java.lang.reflect包下,主要涉及到以下两个类:
Interface InvocationHandler
InvocationHandler接口源码如下:
1 | package java.lang.reflect; |
其中第一个参数proxy代表代理对象;第二个参数method代表被代理的方法;第三个参数代表该方法的参数数组。
Proxy
该类即为动态代理类。
static Object newProxyInstance(ClassLoader loader, Class[] interfaces, InvocationHandler h):返回代理类的一个实例,返回后的代理类可以当作被代理类使用(可使用被代理类在接口中声明过的方法)
JDK动态代理实现步骤
创建一个实现InvocationHandler的类,必须实现invoke方法
创建被代理的类和接口
调用Proxy的静态方法,创建一个代理类
通过代理调用方法
代码实现
下面我们使用动态代理的方式实现静态代理中的例子:
1.定义TimeHandler.java类实现InvocationHandler接口:
1 | public class TimeHandler implements InvocationHandler { |
2.测试类Test.java的实现:
1 | public class Test { |
测试结果如下:
1 | 汽车开始行驶 |
代码详见 PatternTest
参考资料:
王英豪 与接口相关的设计模式(2):代理模式、标识类型模式及常量接口模式
BestWZR 模式的秘密——代理模式
David 模式的秘密——代理模式