一, 动态的核心
动态代理的两个核心,Proxy 和 InvocationHandler,都是Java.lang.reflect包下的接口类;由此也可以看出动态代理是反射的高级应用!
代理与反射的关系如何体现?从代理的建立过程的构建方法及其参数可见一斑:
1,代理最核心的Handler的唯一方法invoke()的核心参数Method method,是一个Class类型,并且在代理调用时,invoke到target对象来返回retval.
2,代理的构造方法参数, 也就是牛X的newProxyInstance()方法的前两个参数, 是目标接口的ClassLoader 和 Interface[ ],也是通过接口Class对象反射get到;
二, 代理的调用方法
Object invoke(Object proxy, Method method,Object[] args) 方法,该方法 在代理实例上处理方法调用并返回结果。
每个代理实例都具有一个关联的调用处理程序。对代理实例调用方法时,将对方法调用进行编码并将其指派到它的调用处理程序的 invoke 方法。
三个参数的意思是:
Object proxy 在其上调用方法的代理实例
Method method 对应于在代理实例上调用的接口方法的Method 实例
Object[] args 包含传入代理实例上方法调用的参数值的对象数组,如果接口方法不使用参数,则为null
三, 代理实例的创建
方法一.1,通过Proxy.getProxyClass()方法创建若干接口的Class文件,
2,通过Class文件可以得到其构造方法
3,最后通过构造方法实例的newInstance方法得到代理实例
方法二.,通过Proxy的newProxyInstance() 方法,直接返回一个带有代理类的指定调用处理程序的代理实例,
它由指定的类加载器定义,并实现指定的接口
该方法的参数列表是:
ClassLoader loader, 指定一个类加载器用于加载指定的接口
Class<?>[] interfaces, 给出需要创建代理实例的接口
InvocationHandlerh, 给出代理实例的调用处理程序实现的接口
显而易见,方法二是对代理逻辑的代码重新封装,有利于编写和管理三大代理构建参数,使大家可以集中精力去处理invoke()方法, target对象,切面信息(Advice),实现AOP编程;
以下为Proxy实例的建立,和进一步AOP编程的应用:
static void clazzProxy()throws Exception{
ClassclazzProxy1 = Proxy.getProxyClass(Collection.class.getClassLoader() ,Collection.class);
System.out.println(clazzProxy1.getName());
System.out.println("---------------constructorslist--------------");
Constructor[]constructors = clazzProxy1.getConstructors();
for(Constructorconstructor : constructors){
Stringname = constructor.getName();
StringBuildersBuilder = new StringBuilder(name);
sBuilder.append('(');
Class[]clazzParams = constructor.getParameterTypes();
for(ClassclazzParam:clazzParams){
sBuilder.append(clazzParam.getName()).append(',');
}
if(clazzParams.length==0)
sBuilder.deleteCharAt(sBuilder.length()-1);
sBuilder.append(')');
System.out.println(sBuilder);
}
System.out.println("-------------begincerate instance --------------");
Constructorconstructor = clazzProxy1.getConstructor(InvocationHandler.class);
classMyInvocationHandler1 implements InvocationHandler{
@Override
publicObject invoke(Object arg0, Method arg1, Object[] arg2)
throwsThrowable {
//TODO Auto-generated method stub
returnnull;
}
};
Collectionproxy1 = (Collection)constructor.newInstance(new MyInvocationHandler1());
System.out.println("xxxxxxxxx"+proxy1.toString());//没有返回值的方法可以调用。有返回值的方法,可以调用
//proxy1.size();//不能调用,因为其没有指定目标
proxy1.clear();//可以调用
Collectionproxy2 = (Collection)constructor.newInstance(new InvocationHandler(){
ArrayListtarget = new ArrayList();
@Override
publicObject invoke(Object proxy, Method method, Object[] args)
throwsThrowable {
longbeginTime = System.currentTimeMillis();
ObjectretVal = method.invoke(target,args);
longendTime = System.currentTimeMillis();
System.out.println(method.getName()+"运行时间"+(endTime-beginTime));
returnretVal;
}
});
proxy2.add("zxx");
proxy2.add("Java01");
proxy2.add("Java");
proxy2.size();
Collection proxy3 = (Collection)Proxy.newProxyInstance(
Collection.class.getClassLoader(),
newClass[]{Collection.class},
newInvocationHandler(){
ArrayList target = new ArrayList();
@Override
publicObject invoke(Object proxy, Method method, Object[] args)
throwsThrowable {
longbeginTime = System.currentTimeMillis();
Object retVal = method.invoke(target,args);
longendTime = System.currentTimeMillis();
System.out.println(method.getName()+"运行时间:"+(endTime-beginTime));
return retVal;
}
}
);
proxy3.add("Java01");
proxy3.add("Java02");
proxy3.add("Java03");
System.out.println(proxy3.size());
}
}