在文章代理模式之静态代理(2)中我们了解到静态代理的运行机制.同时在文章的尾部看到了静态代理的不足.而动态代理可以弥补那些不足,接下来我们将详细的来了解一下动态代理.
在Java的Java.lang.reflect包中提供Proxy类和InvocationHandler.我们可以通过他们两个生成动态的JDK动态代理类或者动态代理对象.
一个类:Proxy(动态生成一个代理对象)必须这个类实现了接口才可以生成代理,如果没有接口的话就不能生成代理.它是通过接口在内存中建立一个类.
一个接口:Interface InvocationHandler.系统生成的每个代理对象都有一个与之相对应的InvocationHandler对象.
具体代码示例:
由于在上文中的接口类和目标类没有进行更改,这里不再赘述.需要更改的是将原来的静态代理中手动添加的代理类删除.取而代之的是一个实现InvocationHandler接口的LogHandler类;
LogHandler类:(在原有的基础之上添加日志功能)
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.text.SimpleDateFormat;
public class LogHandler implements InvocationHandler {
//对目标对象生成代理对象,需要将目标对象传过来.可以使用构造方法和下面的方法
private Object targetObject;
public Object newProxyInstance(Object targetObject){
this.targetObject=targetObject;
//根据传来的对象动态生成代理
//第一个参数是类加载器,需要和目标对象的类加载器一样.
//第二个参数是目标类的所有接口.会根据接口创建出代理类,代理类具有目标类的所有方法,但是方法里没有任何东西.
//第三个参数需要回调InvocationHandler中的invoke方法.也就是实现InvocationHandler的对象.
return Proxy.newProxyInstance(targetObject.getClass().getClassLoader(), targetObject.getClass().getInterfaces(), this);
}
/**
* method表示代理目标的方法,可以动态进行获取
*/
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
//获得当前时间
SimpleDateFormat tempDate = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss");
String datetime = tempDate.format(new java.util.Date());
//方法开始日志记录
System.out.println("startTime: "+datetime+" start-->>"+method.getName());
//方法的相关参数
for(int i=0;i<args.length;i++){
System.out.println(args[i]);
}
//方法的返回值
Object result=null;
try{
//调用目标的方法
result =method.invoke(targetObject, args);
//方法成功日志记录
System.out.println("successTime: "+datetime+" success-->>"+method.getName());
}catch(Exception e){
e.printStackTrace();
//方法失败日志记录
System.out.println("errorTime: "+datetime+" error-->>"+method.getName());
throw e;
}
return result;
}
}
客户端代码:
/**
* 添加一个用户
*/
public class Client {
public static void main(String[] args){
LogHandler logHandler=new LogHandler();
UserManager userManager=(UserManager)logHandler.newProxyInstance(new UserManagerImpl());
userManager.addUser("00001", "刘德华");
}
}
执行结果如下:
startTime: 2012-08-10 16:49:37start-->>addUser
00001
刘德华
UserManagerImpl.addUser()userId-->00001
successTime: 2012-08-10 16:49:37success-->>addUser
使用动态代理,我们可以看到非常好的解耦效果.当然,在我们使用Proxy的时候也不是随意的用的,通常都是为一个指定的目标对象来生成动态代理.这种动态代理在AOP(AspectOrient Program)面向切面编程中称之为AOP代理.AOP代理可以替代目标对象,并且包含目标对象的全部方法,同时在已有方法的基础之上向前,向后加入一些通用处理的方法,例如上例中的日志处理.
通过对动态代理模式的研究和学习可以看到编程的艺术性,同时也在演绎着面向对象的核心思想.