(转)Java动态代理与CGLib代理

时间:2022-03-06 19:56:23
  1. <br>public class UserDAOImpl{
  2. <br><br>    public void save() {
  3. <br>        // TODO Auto-generated method stub
  4. <br>        System.out.println("user saved");
  5. <br>    }
  6. <br>}
  7. <br>//相关配置,省略了一些不相关内容
  8. <br><bean id="userDAO" class="UserDAOImpl">
  9. <br><bean id="userDAOProxy"  class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
  10. <br>    <property name="target">
  11. <br>        <ref local="userDAO" />
  12. <br>    </property>
  13. <br></bean>

测试代码

  1. ApplicationContext ctx =
  2. <br>            new FileSystemXmlApplicationContext("applicationContext.xml");
  3. <br>        UserDAOImpl userDAOImpl =
  4. <br>            (UserDAOImpl)ctx.getBean("userDAOProxy");
  5. <br>        userDAOImpl.save();

上面这种情况下程序可以正常运行,但是如果UserDAOImpl实现了一个接口,其他不变

  1. public class UserDAOImpl implements UserDAO {
  2. <br>
  3. <br>    public void save() {
  4. <br>        // TODO Auto-generated method stub
  5. <br>        System.out.println("user saved");
  6. <br>    }
  7. <br>
  8. <br>}

这种情况下,程序将不能正常运行,会抛出java.lang.ClassCastException异常

理解上面这种情况产生的原因需要了解Spring AOP的实现原理。
Spring 实现AOP是依赖JDK动态代理和CGLIB代理实现的。
以下是JDK动态代理和CGLIB代理简单介绍
    JDK动态代理:其代理对象必须是某个接口的实现,它是通过在运行期间创建一个接口的实现类来完成对目标对象的代理。
    CGLIB代理:实现原理类似于JDK动态代理,只是它在运行期间生成的代理对象是针对目标类扩展的子类。CGLIB是高效的代码生成包,底层是依靠ASM(开源的java字节码编辑类库)操作字节码实现的,性能比JDK强。
Spring是依靠什么来判断采用哪种代理策略来生成AOP代理呢?以下代码就是Spring的判断逻辑

advisedSupport.isOptimize()与advisedSupport.isProxyTargetClass()默认返回都是false,所以在默认情况下目标对象有没有实现接口决定着Spring采取的策略,当然可以设置advisedSupport.isOptimize()或者advisedSupport.isProxyTargetClass()返回为true,这样无论目标对象有没有实现接口Spring都会选择使用CGLIB代理。所以在默认情况下,如果一个目标对象如果实现了接口Spring则会选择JDK动态代理策略动态的创建一个接口实现类(动态代理类)来代理目标对象,可以通俗的理解这个动态代理类是目标对象的另外一个版本,所以这两者之间在强制转换的时候会抛出j ava.lang.ClassCastException。而所以在默认情况下,如果目标对象没有实现任何接口,Spring会选择CGLIB代理, 其生成的动态代理对象是目标类的子类。

以上说的是默认情况下,也可以手动配置一些选项使Spring采用CGLIB代理。

org.springframework.transaction.interceptor.TransactionProxyFactoryBean是org.springframework.aop.framework. ProxyConfig的子类,所以可以参照ProxyConfig里的一些设置如下所示,将optimize和proxyTargetClass任意一个设置为true都可以强制Spring采用CGLIB代理。

  1. //相关配置,省略了一些不相关内容
  2. <br><bean id="userDAO" class="UserDAOImpl">
  3. <br><bean id="userDAOProxy"  class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
  4. <br>    <property name="target">
  5. <br>        <ref local="userDAO" />
  6. <br>    </property>
  7. <br>    <property name="optimize">
  8. <br>        <value>true</value>
  9. <br>    </property>
  10. <br>    <property name="proxyTargetClass">
  11. <br>        <value>true</value>
  12. <br>    </property>
  13. <br></bean>

使用CGLIB代理也就不会出现前面提到的ClassCastException问题了,

也可以在性能上有所提高,但是也有它的弊端,Spring doc原文解释如下optimization will usually mean that advice changes won't take effect after a proxy has been created. For this reason, optimization  is disabled by default。