java的动态代理使用Proxy.newInstance(classloader,interfaces,invocationHandler)方法
该方法进行以下操作:
用classloader作为Proxy0的定义类加载器在内存中加载一个Proxy0类,并实现interfaces的所有接口,同时将invocationHandler传进去Proxy中
也就是说,将接口定义的方法委托给Proxy,Proxy实现了接口所有的方法;
Proxy内部使用InvocationHandler调用所有接口的方法,而该handler就是我们编写的代理实现逻辑
ps:Proxy0类字节码的获取可以修改jdk的classloader的defineclass方法,在加载字节码时临时将byte数组保存到文件中。或者自己实现一个classloader传入Proxy.newInstance方法
另外。springaop中有还有一个实现的方式叫cglib
cglib是通过继承类的形式,动态生成一个子类。
但是继承类的方式不能aop拦截父类中final方法
如果想aop拦截父类的final方法,只能使用类似asm字节码的暴力手段了。
该方法进行以下操作:
用classloader作为Proxy0的定义类加载器在内存中加载一个Proxy0类,并实现interfaces的所有接口,同时将invocationHandler传进去Proxy中
interface Bussiness {
public void doBussiness();
}
public class BussinessImpl implements Bussiness {
@Override
public void doBussiness() {
System.out.println("i do bussiness");
}
}
public class BussinessInf implements InvocationHandler {
private Object obj;
public BussinessInf(Object obj) {
this.obj = obj;
}
@Override
public Object invoke(Object proxy, Method m, Object[] args)
throws Throwable {
System.out.println(proxy.getClass().getName());
System.out.println("do something before bussiness");
return m.invoke(obj, args);
}
}
public class TestMain {
public static void main(String[] args) {
Bussiness b = new BussinessImpl();
BussinessInf proxy = new BussinessInf(b);
Bussiness b1 = (Bussiness)
Proxy.newProxyInstance(b.getClass().getClassLoader() , b.getClass().getInterfaces(),proxy);
System.out.println(b1.getClass().getName());
// proxytest.$Proxy0
b1.doBussiness();
}
}
也就是说,将接口定义的方法委托给Proxy,Proxy实现了接口所有的方法;
Proxy内部使用InvocationHandler调用所有接口的方法,而该handler就是我们编写的代理实现逻辑
ps:Proxy0类字节码的获取可以修改jdk的classloader的defineclass方法,在加载字节码时临时将byte数组保存到文件中。或者自己实现一个classloader传入Proxy.newInstance方法
另外。springaop中有还有一个实现的方式叫cglib
cglib是通过继承类的形式,动态生成一个子类。
但是继承类的方式不能aop拦截父类中final方法
如果想aop拦截父类的final方法,只能使用类似asm字节码的暴力手段了。