Spring AOP 代理机制:proxy-target-class 决定 用 CGlib 还是 JDK AOP 来生成代理

参考
https://blog.csdn.net/bigtree_3721/article/details/50774044
http://www.cnblogs.com/hustyangli/archive/2008/09/01/1281319.html

Spring Framework reference 2.0.5 参考手册中文版 6.6. 代理机制 中是这样写的

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

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

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

  • 无法通知(advice)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) {   

                throw new 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 {   

            return new JdkDynamicAopProxy(advisedSupport);   

        }   

    }

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

所以在默认情况下,如果一个目标对象如果实现了接口,Spring则会选择JDK动态代理策略动态的创建一个接口实现类(动态代理类)来代理目标对象,可以通俗的理解这个动态代理类是目标对象的另外一个版本

所以在默认情况下,如果目标对象没有实现任何接口,Spring会选择CGLIB代理其生成的动态代理对象是目标类的子类

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


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

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

二者有如下的区别:

Aspect默认情况下不用实现接口但对于目标对象(UserManagerImpl.java),在默认情况下必须实现接口,如果没有实现接口必须引入CGLIB库,我们可以通过Advice中添加一个JoinPoint参数,这个值会由spring自动传入,从JoinPoint中可以取得参数值、方法名等等

  • 如果目标对象实现了接口,默认情况下会采用JDK的动态代理实现AOP
  • 如果目标对象实现了接口,可以强制使用CGLIB实现AOP
  • 如果目标对象没有实现接口必须采用CGLIB库,spring会自动在JDK动态代理和CGLIB之间转换

如何强制使用CGLIB实现AOP?

  • 添加CGLIB库,SPRING_HOME/cglib/*.jar
  • 在spring配置文件中加入< aop:aspectj-autoproxy proxy-target-class=“true”/>

JDK动态代理和CGLIB字节码生成的区别?

  • JDK动态代理只能对实现了接口的类生成代理,而不能针对类。原理是通过动态代理,生成接口的实例类和对象,从而进行对目标实例的代理。
  • CGLIB是针对类实现代理,主要是对指定的类生成一个子类,覆盖其中的方法
    因为是继承,所以目标类或方法最好不要声明成final

你可能感兴趣的:(Spring)