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字节码的暴力手段了。
本文深入探讨了Java动态代理的实现原理及使用方法,包括Proxy.newInstance方法的应用,以及如何通过自定义InvocationHandler实现业务逻辑。同时,文章对比了Spring AOP中的CGlib实现方式,并解释了其在拦截父类final方法时的局限性,提供了针对不同场景的选择建议。
1784

被折叠的 条评论
为什么被折叠?



