java动态代理与反射
引言:
java的代理分为静态代理与动态代理,静态代理有其固有缺点,即代码量大,缺乏灵活性。故我们主要关注的还是动态代理。而动态代理是基于java反射机制实现的一种代理模式。
动态代理的用处:动态代理的用处很广。比如,我们常用的spring的AOP,即是通过动态代理技术来实现的。
要理解动态代理,首先需要了解java反射的基本概念及API方法。在这里假设你已经知道java反射的基础知识,在这个前提下,我们通过一个例子程序,即可说明动态代理的实现原理:
interface Manager {
public void modify1();
public void modify2();
}
class ManagerImpl implements Manager {
public void modify1() {
System.out.println("*******ManagerImpl 的 modify()1方法被调用*******");
}
public void modify2() {
System.out.println("*******ManagerImpl 的 modify()2方法被调用*******");
}
}
class ManagerImpl2 implements Manager {
public void modify1() {
System.out.println("*******ManagerImpl2 的 modify()1方法被调用*******");
}
public void modify2() {
System.out.println("*******ManagerImpl2 的 modify()2方法被调用*******");
}
}
class BusinessHandler implements InvocationHandler {
private Object object = null;
public BusinessHandler(Object object) {
this.object = object;
}
@Override
public Object invoke(Object proxy, Method method, Object[] args)
throws Throwable {
System.out.println("do something before method");
Object ret = method.invoke(this.object, args);
System.out.println("do something after method");
return ret;
}
}
public class Client {
public static void main(String[] args) {
// 元对象(被代理对象)
ManagerImpl managerImpl = new ManagerImpl();
ManagerImpl2 managerImpl2 = new ManagerImpl2();
// 业务代理类
BusinessHandler securityHandler = new BusinessHandler(managerImpl);
// 获得代理类($Proxy0 extends Proxy implements Manager)的实例.
Manager managerProxy = (Manager) Proxy.newProxyInstance(managerImpl
.getClass().getClassLoader(), managerImpl.getClass()
.getInterfaces(), securityHandler);
System.out.println("——————");
//通过反射,可以知晓,在内存中实际生成了一个$Proxy0的类,亦可通过反射看出这个类的代码组成
//这里可以通过运行结果证明managerProxy是Proxy的一个实例,这个实例实现了Manager接口
System.out.println(managerProxy instanceof Proxy);
//这里可以看出managerProxy的Class类是$Proxy0,这个$Proxy0类继承了Proxy,实现了Manager接口
System.out.println("managerProxy的Class类是:"+managerProxy.getClass().toString());
System.out.print("managerProxy中的属性有:");
Field[] field=managerProxy.getClass().getDeclaredFields();
for(Field f:field){
System.out.print(f.getName()+", ");
}
System.out.print("\n"+"managerProxy中的方法有:");
Method[] method=managerProxy.getClass().getDeclaredMethods();
for(Method m:method){
System.out.print(m.getName()+", ");
}
System.out.println("\n"+"managerProxy的父类是:"+managerProxy.getClass().getSuperclass());
System.out.print("\n"+"managerProxy实现的接口是:");
Class>[] interfaces=managerProxy.getClass().getInterfaces();
for(Class> i:interfaces){
System.out.print(i.getName()+", ");
}
System.out.println();
System.out.println("——————");
managerProxy.modify1();
managerProxy.modify2();
System.out.println("——————");
// 业务代理类,用以演示动态代理的优势:不用增加格外代码,只需增加另一个代理处理类,然后绑定到一个新的InvocationHandler实例上,即可实现扩充
securityHandler = new BusinessHandler(managerImpl2);
// 获得代理类($Proxy0 extends Proxy implements Manager)的实例.
managerProxy = (Manager) Proxy.newProxyInstance(managerImpl2
.getClass().getClassLoader(), managerImpl2.getClass()
.getInterfaces(), securityHandler);
managerProxy.modify1();
managerProxy.modify2();
}
}
执行结果:
——————
true
managerProxy的Class类是:class testReflect_10.$Proxy0
managerProxy中的属性有:m4, m1, m0, m3, m2,
managerProxy中的方法有:equals, toString, hashCode, modify1, modify2,
managerProxy的父类是:class java.lang.reflect.Proxy
managerProxy实现的接口是:testReflect_10.Manager,
——————
do something before method
*******ManagerImpl 的 modify()1方法被调用*******
do something after method
do something before method
*******ManagerImpl 的 modify()2方法被调用*******
do something after method
——————
do something before method
*******ManagerImpl2 的 modify()1方法被调用*******
do something after method
do something before method
*******ManagerImpl2 的 modify()2方法被调用*******
do something after method
说明:
1、InvocationHandler是个代理处理类的父类(接口),只定义了一个invoke方法。BusinessHandler继承了该接口,并实现了invoke方法;
2、securityHandler = new BusinessHandler(managerImpl):实例化一个代理处理类,并传入一个要代理的真实“对象”:managerImpl。
3、(Manager) Proxy.newProxyInstance(managerImpl
.getClass().getClassLoader(), managerImpl.getClass()
.getInterfaces(), securityHandler) 方法:创建了一个$Proxy0类的实例(不要问我是怎么知道的,代码里已经很清楚的看到,是可以通过反射打印出来的),这个实例继承了Proxy类,也实现了要代理的那个Manager接口,同时,还传入一个BusinessHandler的实例。
4、重点来了,managerProxy.modify1(),这句代码执行的时候,为什么会调用BusinessHandler的 invoke()方法?答案:也是通过反射,让我看到了那个$Proxy0类的代码(你可以通过反射的各种方法去将这个类的所有信息都可以打印出来)。其实网上也有很多同学已经帮我们去辛辛苦苦扣下来了$Proxy0类的代码,如是:
import java.lang.reflect.InvocationHandler;
import java.lang.reflect.Method;
import java.lang.reflect.Proxy;
import java.lang.reflect.UndeclaredThrowableException;
public final class $Proxy0 extends Proxy implements Manager{
private static Method m4;
private static Method m1;
private static Method m0;
private static Method m3;
private static Method m2;
static {
try {
m4 =Class.forName("java.lang.Object").getMethod("equals",
new Class[] { Class.forName("java.lang.Object") });
m1 =Class.forName("java.lang.Object").getMethod("toString",
new Class[0]);
m0 =Class.forName("java.lang.Object").getMethod("hashCode",
new Class[0]);
m3 =Class.forName("com.ml.test.Manager").getMethod("modify1",
new Class[0]);
m2 =Class.forName("com.ml.test.Manager").getMethod("modify2",
new Class[0]);
} catch (NoSuchMethodExceptionnosuchmethodexception) {
throw newNoSuchMethodError(nosuchmethodexception.getMessage());
} catch(ClassNotFoundException classnotfoundexception) {
throw newNoClassDefFoundError(classnotfoundexception.getMessage());
}
} //static
public $Proxy0(InvocationHandler invocationhandler) {
super(invocationhandler);
}
@Override
public final boolean equals(Object obj) {
try {
return ((Boolean) super.h.invoke(this, m4, new Object[] { obj })).booleanValue();
} catch (Throwable throwable) {
throw newUndeclaredThrowableException(throwable);
}
}
@Override
public final int hashCode() {
try {
return ((Integer)super.h.invoke(this, m0, null)).intValue();
} catch (Throwable throwable) {
throw newUndeclaredThrowableException(throwable);
}
}
public final void modify1() {
try {
super.h.invoke(this, m3, null);
return;
} catch (Error e) {
} catch (Throwable throwable) {
throw new UndeclaredThrowableException(throwable);
}
}
public final void modify2() {
try {
super.h.invoke(this, m2, null);
return;
} catch (Error e) {
} catch (Throwable throwable) {
throw new UndeclaredThrowableException(throwable);
}
}
@Override
public final String toString() {
try {
return (String) super.h.invoke(this, m1,null);
} catch (Throwable throwable) {
throw new UndeclaredThrowableException(throwable);
}
}
}
看到关键代码没?这个:
public final void modify1() {
try {
super.h.invoke(this, m3, null);
return;
明白了吧?代理类每次执行方法时,实际就是通过BusinessHandler这个真正的代理处理类的invoke方法来执行的。invoke方法的:method.invoke(this.object, args)这个方法,是java的反射技术,通过这个方法其实真正执行了managerImpl的modify1()这些方法。
——————这下明白了吧,动态代理的实现步骤:
1、继承并实现InvocationHandler接口及其invoke方法,这个方法体的实现其实也比较固定,就如例子程序那样;
2、通过Proxy的newProxyInstance方法创建一个代理实例,这个创建方法也会把真正处理方法执行的InvocationHandler绑定在一块。
3、再用第二步的proxy实例执行方法时,实际是调用了InvocationHandler的invoke方法,最终执行了真实的managerImpl的modify1()方法。
最后,我再画一个图吧:
希望对大家理解动态代理有所帮助!谢谢。