Spring强制使用CGLIB代理事务

来源:互联网 发布:python getopt usage 编辑:程序博客网 时间:2024/05/29 18:22
代理的核心就是拦截方法调用,并在需要的时候执行匹配某方法的通知链。 

和CGLIB不同的是,JDK代理只能代理接口,不能代理类。 

使用JDK代理时,如何处理一个特定的方法调用的决定是在程序运行时做出的,也就是在每次方法被调用时。使用CGLIB代理可以边开这种处理方法,CGLIB会在运行中随时为代理创建新类的字节码,并尽可能的重用已经生成的类的字节码。 


Spring1.2:
将事务代理工厂[TransactionProxyFactoryBean] 或 自动代理拦截器[BeanNameAutoProxyCreator]
的 proxyTargetClass 属性,设置为true,则使用CGLIB代理,此属性默认为false,使用JDK动态代理.
以下引用 Spring Framework reference 2.0.5:
Spring2.0:

Spring AOP部分使用JDK动态代理或者CGLIB来为目标对象创建代理。(建议尽量使用JDK的动态代理)

如果被代理的目标对象实现了至少一个接口,则会使用JDK动态代理。所有该目标类型实现的接口都将被代理。若该目标对象没有实现任何接口,则创建一个CGLIB代理。

如果你希望强制使用CGLIB代理,(例如:希望代理目标对象的所有方法,而不只是实现自接口的方法)那也可以。但是需要考虑以下问题:

无法通知(advise)Final 方法,因为他们不能被覆写。 
你需要将CGLIB 2二进制发行包放在classpath下面,与之相较JDK本身就提供了动态代理 
强制使用CGLIB代理需要将 |aop:config| 的 proxy-target-class 属性设为true:

|aop:config proxy-target-class="true"|
...
|/aop:config|

当需要使用CGLIB代理和@AspectJ自动代理支持,请按照如下的方式设置 |aop:aspectj-autoproxy| 的 proxy-target-class 属性:

|aop:aspectj-autoproxy proxy-target-class="true"/|

而实际使用的过程中才会发现细节问题的差别,The devil is in the detail.JDK动态代理:其代理对象必须是某个接口的实现,它是通过在运行期间创建一个接口的实现类来完成对目标对象的代理。
CGLIB代理:实现原理类似于JDK动态代理,只是它在运行期间生成的代理对象是针对目标类扩展的子类。CGLIB是高效的代码生成包,底层是依靠ASM(开源的java字节码编辑类库)操作字节码实现的,性能比JDK强。
Spring是依靠什么来判断采用哪种代理策略来生成AOP代理呢?以下代码就是Spring的判断逻辑  

    //org.springframework.aop.framework.DefaultAopProxyFactory   
    
//参数AdvisedSupport 是Spring AOP配置相关类   
    public AopProxy createAopProxy(AdvisedSupport advisedSupport)   
            
throws AopConfigException {   
        
//在此判断使用JDK动态代理还是CGLIB代理   
        if (advisedSupport.isOptimize() || advisedSupport.isProxyTargetClass()   
                
|| hasNoUserSuppliedProxyInterfaces(advisedSupport)) {   
            
if (!cglibAvailable) {   
                
thrownew AopConfigException(   
                        
"Cannot proxy target class because CGLIB2 is not available. "  
                                
+"Add CGLIB to the class path or specify proxy interfaces.");   
              }   
            
return CglibProxyFactory.createCglibProxy(advisedSupport);   
          } 
else {   
            
returnnew JdkDynamicAopProxy(advisedSupport);   
          }   
      }  

advisedSupport.isOptimize()与advisedSupport.isProxyTargetClass()默认返回都是false,所以在默认情况下目标对象有没有实现接口决定着Spring采取的策略,当然可以设置advisedSupport.isOptimize()或者advisedSupport.isProxyTargetClass()返回为true,这样无论目标对象有没有实现接口Spring都会选择使用CGLIB代理。

所以在默认情况下,如果一个目标对象如果实现了接口Spring则会选择JDK动态代理策略动态的创建一个接口实现类(动态代理类)来代理目标对象,可以通俗的理解这个动态代理类是目标对象的另外一个版本,所以这两者之间在强制转换的时候会抛出java.lang.ClassCastException。而所以在默认情况下,如果目标对象没有实现任何接口,Spring会选择CGLIB代理, 其生成的动态代理对象是目标类的子类。

上说的是默认情况下,也可以手动配置一些选项使Spring采用CGLIB代理。 
org.springframework.transaction.interceptor.TransactionProxyFactoryBean是org.springframework.aop.framework. ProxyConfig的子类,所以可以参照ProxyConfig里的一些设置如下所示,将optimize和proxyTargetClass任意一个设置为true都可以强制Spring采用CGLIB代理。

如果当需要使用CGLIB代理和@AspectJ自动代理支持,请按照如下的方式设置 |aop:aspectj-autoproxy| 的 proxy-target-class 属性: 
|aop:aspectj-autoproxy proxy-target-class="true"/|

这样使用CGLIB代理也就不会出现前面提到的ClassCastException问题了,也可以在性能上有所提高,关键是对于代理对象是否继承接口可以统一使用。





两种类型AOP:静态AOP和动态AOP。 

静态代理: 
代理对象与被代理对象必须实现同一个接口。
 
demo:

view plaincopy to clipboardprint?
  1. package cn.partner4java.proxy.staticproxy;  
  2.   
  3.   
  4. /** 
  5. * 静态代理,统一接口 
  6. * @author partner4java 
  7. * 
  8. */  
  9. public interface IHello {  
  10.     /** 
  11.      * 可以带来的统一方法 
  12.      * @param name 
  13.      */  
  14.     public void hello(String name);  
  15. }  
view plaincopy to clipboardprint?
  1. package cn.partner4java.proxy.staticproxy;  
  2.   
  3. /** 
  4. * 被代理的对象,需要借助代理对象加入日志 
  5. * @author partner4java 
  6. * 
  7. */  
  8. public class HelloSpeaker implements IHello {  
  9.   
  10.     public void hello(String name) {  
  11.         System.out.println("Hello " + name);  
  12.     }  
  13.   
  14. }  
view plaincopy to clipboardprint?
  1. package cn.partner4java.proxy.staticproxy;  
  2.   
  3. /** 
  4. * 代理对象,给被代理对象添加日志 
  5. */  
  6. public class HelloProxy implements IHello {  
  7.       
  8.     private IHello iHello;  
  9.   
  10.     public HelloProxy(IHello iHello) {  
  11.         super();  
  12.         this.iHello = iHello;  
  13.     }  
  14.   
  15.   
  16.     public void hello(String name) {  
  17.         System.out.println("记录日志");  
  18.         iHello.hello(name);  
  19.     }  
  20.   
  21. }  
view plaincopy to clipboardprint?
  1. package cn.partner4java.proxy.staticproxy;  
  2.   
  3. /** 
  4. * 调用 
  5. * @author partner4java 
  6. * 
  7. */  
  8. public class ProxyDemo {  
  9.   
  10.     public static void main(String[] args) {  
  11.         IHello iHello = new HelloProxy(new HelloSpeaker());  
  12.         iHello.hello("long");  
  13.     }  
  14.   
  15. }  





动态代理: 
动态代理区别于静态带来实现的地方在于织入过程是在运行时动态进行的。自己实现一般实现java.lang.reflect.InvocationHandler接口。 
例子:
view plaincopy to clipboardprint?
  1. package cn.partner4java.proxy.dynamicproxy;  
  2.   
  3.   
  4. public interface IHello {  
  5.     public void hello(String name);  
  6. }  
view plaincopy to clipboardprint?
  1. package cn.partner4java.proxy.dynamicproxy;  
  2.   
  3. /** 
  4. * 被代理的对象,需要借助代理对象加入日志 
  5. * @author partner4java 
  6. * 
  7. */  
  8. public class HelloSpeaker implements IHello {  
  9.   
  10.     public void hello(String name) {  
  11.         System.out.println("Hello " + name);  
  12.     }  
  13.   
  14. }  
view plaincopy to clipboardprint?
  1. package cn.partner4java.proxy.dynamicproxy;  
  2.   
  3. import java.lang.reflect.InvocationHandler;  
  4. import java.lang.reflect.Method;  
  5. import java.lang.reflect.Proxy;  
  6.   
  7. /** 
  8. * 动态代理对象 
  9. * @author partner4java 
  10. * 
  11. */  
  12. public class LogHandler implements InvocationHandler {  
  13.   
  14.     private Object delegate;  
  15.       
  16.     public Object bind(Object delegate){  
  17.         this.delegate = delegate;  
  18.         return Proxy.newProxyInstance(delegate.getClass().getClassLoader(),   
  19.                 delegate.getClass().getInterfaces(), this);  
  20.     }  
  21.     /** 
  22.      * 代理对象,这里面还可以改变原有的方法 
  23.      */  
  24.     public Object invoke(Object proxy, Method method, Object[] args)  
  25.             throws Throwable {  
  26.         Object result = null;  
  27.         try {  
  28.             System.out.println("添加日志");  
  29.             result = method.invoke(delegate, args);  
  30.         } catch (Exception e) {  
  31.             e.printStackTrace();  
  32.         }  
  33.           
  34.         return null;  
  35.     }  
  36.   
  37. }  
view plaincopy to clipboardprint?
  1. package cn.partner4java.proxy.dynamicproxy;  
  2.   
  3. /** 
  4. * 测试 
  5. * @author partner4java 
  6. * 
  7. */  
  8. public class ProxyDemo {  
  9.     public static void main(String[] args) {  
  10.         LogHandler logHandler = new LogHandler();  
  11.         IHello iHello = (IHello) logHandler.bind(new HelloSpeaker());  
  12.         iHello.hello("long");  
  13.     }  
  14. }  


------------------------------------------------------------------ 


利用ProxyFactory连接CGLIB简单实现AOP: 
加入包aopalliance.jar\cglib-nodep-2.1_3.jar 
demo:
view plaincopy to clipboardprint?
  1. package cn.partner4java.proxy.proxyfactory;  
  2.   
  3. /** 
  4. * 被代理的对象 
  5. * @author partner4java 
  6. * 
  7. */  
  8. public class MessageWriter {  
  9.     public void writeMessage(){  
  10.         System.out.println("world!");  
  11.     }  
  12. }  
view plaincopy to clipboardprint?
  1. package cn.partner4java.proxy.proxyfactory;  
  2.   
  3. import org.aopalliance.intercept.MethodInterceptor;  
  4. import org.aopalliance.intercept.MethodInvocation;  
  5.   
  6. /** 
  7. * 装饰者<br/> 
  8. * MethodInterceptor接口是对方法调用连接点实现包围通知的AOP联盟标准接口 
  9. * @author partner4java 
  10. * 
  11. */  
  12. public class MessageDecorator implements MethodInterceptor{  
  13.   
  14.     public Object invoke(MethodInvocation invocation) throws Throwable {  
  15.         System.out.print("Hello ");  
  16.         Object retVal = invocation.proceed();  
  17.         return retVal;  
  18.     }  
  19.   
  20. }  
view plaincopy to clipboardprint?
  1. package cn.partner4java.proxy.proxyfactory;  
  2.   
  3. import org.springframework.aop.framework.ProxyFactory;  
  4.   
  5. /** 
  6. * 调用组装 
  7. * 这里最重要的部分是我们使用ProxyFactory来创建一个目标对象代理,同时织入通知  
  8. * @author partner4java 
  9. * 
  10. */  
  11. public class HelloWorldWeaver {  
  12.   
  13.     public static void main(String[] args) {  
  14.         //目标  
  15.         MessageWriter target = new MessageWriter();  
  16.           
  17.         //create the proxy  
  18.         ProxyFactory proxyFactory = new ProxyFactory();  
  19.           
  20.         proxyFactory.addAdvice(new MessageDecorator());  
  21.         proxyFactory.setTarget(target);  
  22.           
  23.         //获取返回被代理的目标  
  24.         MessageWriter proxy = (MessageWriter) proxyFactory.getProxy();  
  25.           
  26.         target.writeMessage();  
  27.         System.out.println("---");  
  28.         proxy.writeMessage();  
  29. //      后台打印:  
  30. //      world!  
  31. //      ---  
  32. //      World world!  
  33.     }  
  34.   
  35. }  




原创粉丝点击