`
dingchd
  • 浏览: 16065 次
  • 性别: Icon_minigender_1
  • 来自: 北京
社区版块
存档分类
最新评论

java动态代理原理简单描述

    博客分类:
  • java
 
阅读更多
java的动态代理使用Proxy.newInstance(classloader,interfaces,invocationHandler)方法

该方法进行以下操作:
用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字节码的暴力手段了。
分享到:
评论

相关推荐

Global site tag (gtag.js) - Google Analytics