posts - 110, comments - 101, trackbacks - 0, articles - 7
  BlogJava :: 首页 :: 新随笔 :: 联系 :: 聚合  :: 管理
这篇文章试验了JDK动态代理与CGLIB动态代理。从Spring的AOP框架介绍中得知对于使用接口的类,Spring使用JDK 动态代理(原来做项目中试图从Bean强制转换为实现类,结果报错,原来是这么回事),没有接口的就使用别的AOP框架aspectj,但这些都是依赖于 Java字节码工具ASM生成一个原类的新类,调用Callback

但是JDK动态代理为什么必须使用接口一直很疑惑,难道原理不是像ASM一样修改字节码吗?带着这个疑问,开始看JDK的Proxy代码。使用JDK动态代理的代码代码。

Java代码 复制代码 收藏代码
  1. ITestBean tb = (ITestBean) Proxy.newProxyInstance(tb.getClass().getClassLoader(), tb.getClass().getInterfaces(), new TestBeanHander(tb));


于是从创建代理函数看起,即public static Object newProxyInstance(ClassLoader loader,
Class<?>[] interfaces, InvocationHandler h)
throws IllegalArgumentException ,

通过源码可以看到,这个类第一步生成一个代理类(注意,这里的参数就是接口列表),

Java代码 复制代码 收藏代码
  1. Class cl = getProxyClass(loader, interfaces);


然后通过代理类找到构造参数为InvocationHandler的构造函数并生成一个新类。

Java代码 复制代码 收藏代码
  1. Constructor cons = cl.getConstructor(constructorParams);//这个有用,在后面细说
  2. return (Object) cons.newInstance(new Object[] { h });


接口起什么作用呢,于是又看getProxyClass方法的代码,这个源码很长,就不细说了。大致分为三段:

第一:验证

第二:缓存创建新类的结构,如果创建过,则直接返回。(注意:这里的KEY就是接口列表)

第三:如果没有创建过,则创建新类

创建代码如下
Java代码 复制代码 收藏代码
  1. long num;
  2. //获得代理类数字标识
  3. synchronized (nextUniqueNumberLock) {
  4. num = nextUniqueNumber++;
  5. }
  6. //获得创建新类的类名$Proxy,包名为接口包名,但需要注意的是,如果有两个接口而且不在同一个包下,也会报错
  7. String proxyName = proxyPkg + proxyClassNamePrefix + num;
  8. //调用class处理文件生成类的字节码,根据接口列表创建一个新类,这个类为代理类,
  9. byte[] proxyClassFile = ProxyGenerator.generateProxyClass(
  10. proxyName, interfaces);
  11. //通过JNI接口,将Class字节码文件定义一个新类
  12. proxyClass = defineClass0(loader, proxyName,
  13. proxyClassFile, 0, proxyClassFile.length);



根据前面的代码Constructor cons = cl.getConstructor(constructorParams);

可以猜测到接口创建的新类proxyClassFile 不管采用什么接口,都是以下结构

Java代码 复制代码 收藏代码
  1. public class $Proxy1 extends Proxy implements 传入的接口{
  2. }


生成新类的看不到源代码,不过猜测它的执行原理很有可能是如果类是Proxy的子类,则调用InvocationHandler进行方法的Invoke

到现在大家都应该明白了吧,JDK动态代理的原理是根据定义好的规则,用传入的接口创建一个新类,这就是为什么采用动态代理时为什么只能用接口引用指向代理,而不能用传入的类引用执行动态类。

cglib采用的是用创建一个继承实现类的子类,用asm库动态修改子类的代码来实现的,所以可以用传入的类引用执行代理类

JDK动态代理与CGLIB对比如下:

//JDK动态代理测试代码

Java代码 复制代码 收藏代码
  1. ITestBean tb = new TestBean();
  2. tb = (ITestBean) Proxy.newProxyInstance(tb.getClass().getClassLoader(), tb.getClass().getInterfaces(), new TestBeanHander(tb));//这句用接口引用指向,不会报错
  3. TestBean tmp = (TestBean) tb;//强制转换为实现类,将抛出类强制转换异常


//CGLIB测试代码
Java代码 复制代码 收藏代码
  1. TestProxy tp = new TestProxy();
  2. tb = (ITestBean) tp.getProxy(TestBean.class);
  3. tmp = (TeatBean) tb;//强制转换为实现类,不会抛出异常

补充说明,如果在实现类中,接口定义的方法互相调用不会在调用InvocationHandler的invoke方法,JDK动态代理应该不是嵌入到Java的反射机制中,而是在反射机制上的一个调用。


应用举例如下:

JDK动态代理的简单使用示例:


Java代码 复制代码 收藏代码
  1. package com.proxy;
  2. public class ForumServiceImpl implements ForumService{
  3. public void removeTopic(int topicId){
  4. System.out.println("模拟删除记录"+topicId);
  5. try{
  6. Thread.currentThread().sleep(20);
  7. }catch(Exception e){
  8. throw new RuntimeException(e);
  9. }
  10. }
  11. public void removeForum(int forumId){
  12. System.out.println("模拟删除记录"+forumId);
  13. try{
  14. Thread.currentThread().sleep(20);
  15. }catch(Exception e){
  16. throw new RuntimeException(e);
  17. }
  18. }
  19. }


创建一个实现java.lang.reflect.InvocationHandler 接口的代理类,如:
Java代码 复制代码 收藏代码
  1. import java.lang.reflect.InvocationHandler;
  2. import java.lang.reflect.Method;
  3. public class PerformanceHandler implements InvocationHandler{
  4. private Object target; //要进行代理的业务类的实例
  5. public PerformanceHandler(Object target){
  6. this.target = target;
  7. }
  8. //覆盖java.lang.reflect.InvocationHandler的方法invoke()进行织入(增强)的操作
  9. //在实际应用中, 这里会引用一个Intercepter类来做处理。 然后Intercepter就可以独立发展
  10. public Object invoke(Object proxy, Method method, Object[] args)
  11. throws Throwable{
  12. System.out.println("Object target proxy:"+target);
  13. System.out.println("模拟代理加强的方法...");
  14. Object obj = method.invoke(target, args); //调用目标业务类的方法
  15. System.out.println("模拟代理加强的方法执行完毕...");
  16. return obj;
  17. }
  18. }



用java.lang.reflect.Proxy.newProxyInstance()方法创建动态实例来调用代理实例的方法:
Java代码 复制代码 收藏代码
  1. import java.lang.reflect.Proxy;
  2. public class TestForumService {
  3. public static void main(String args[]){
  4. ForumService target = new ForumServiceImpl();//要进行代理的目标业务类
  5. PerformanceHandler handler = new PerformanceHandler(target);//用代理类把目标业务类进行编织
  6. //创建代理实例,它可以看作是要代理的目标业务类的加多了横切代码(方法)的一个子类
  7. ForumService proxy = (ForumService)Proxy.newProxyInstance(
  8. target.getClass().getClassLoader(),
  9. target.getClass().getInterfaces(), handler);
  10. proxy.removeForum(10);
  11. proxy.removeTopic(20);
  12. }
  13. }



CGLib动态代理示例:


创建一个实现net.sf.cglib.proxy.MethodInterceptor接口的实例来为目标业务类加入进行代理时要进行的操作或增强:

Java代码 复制代码 收藏代码
  1. import java.lang.reflect.Method;
  2. import net.sf.cglib.proxy.MethodProxy;
  3. import net.sf.cglib.proxy.Enhancer;
  4. import net.sf.cglib.proxy.MethodInterceptor;
  5. /**
  6. *CGlib采用非常底层的字节码技术,可以为一个类创建子类,
  7. 并在子类中采用方法拦截技术拦截父类方法的调用,并顺势进行增强,即是织入横切逻辑
  8. * @author tufu
  9. */
  10. public class CglibProxy implements MethodInterceptor{
  11. private Enhancer enhancer = new Enhancer();
  12. //覆盖MethodInterceptor接口的getProxy()方法,设置
  13. public Object getProxy(Class clazz){
  14. enhancer.setSuperclass(clazz); //设者要创建子类的类
  15. enhancer.setCallback(this); //设置回调的对象
  16. return enhancer.create(); //通过字节码技术动态创建子类实例,
  17. }
  18. public Object intercept(Object obj,Method method,Object[] args,
  19. MethodProxy proxy) throws Throwable {
  20. System.out.println("模拟代理增强方法");
  21. //通过代理类实例调用父类的方法,即是目标业务类方法的调用
  22. Object result = proxy.invokeSuper(obj, args);
  23. System.out.println("模拟代理增强方法结束");
  24. return result;
  25. }
  26. }


通过java.lang.reflect.Proxy的getProxy()动态生成目标业务类的子类,即是代理类,再由此得到代理实例:
Java代码 复制代码 收藏代码
  1. import com.proxy.ForumServiceImpl;
  2. import java.lang.reflect.Proxy;
  3. public class TestCglibProxy {
  4. public static void main(String args[]){
  5. CglibProxy proxy = new CglibProxy();
  6. //动态生成子类的方法创建代理类
  7. ForumServiceImpl fsi =
  8. (ForumServiceImpl)proxy.getProxy(ForumServiceImpl.class);
  9. fsi.removeForum(10);
  10. fsi.removeTopic(2);
  11. }
  12. }


总结下Spring的AOP运用的设计模式 , AOP 主要利用代理模式, 然后依赖通知(本人认为是策略模式)来实现AOP。 这样通知就可以独立发展。

只有注册用户登录后才能发表评论。


网站导航: