- <br>public class UserDAOImpl{
- <br><br> public void save() {
- <br> // TODO Auto-generated method stub
- <br> System.out.println("user saved");
- <br> }
- <br>}
- <br>//相关配置,省略了一些不相关内容
- <br><bean id="userDAO" class="UserDAOImpl">
- <br><bean id="userDAOProxy" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
- <br> <property name="target">
- <br> <ref local="userDAO" />
- <br> </property>
- <br></bean>
測试代码
- ApplicationContext ctx =
- <br> new FileSystemXmlApplicationContext("applicationContext.xml");
- <br> UserDAOImpl userDAOImpl =
- <br> (UserDAOImpl)ctx.getBean("userDAOProxy");
- <br> userDAOImpl.save();
上面这样的情况下程序能够正常执行,可是假设UserDAOImpl实现了一个接口,其它不变
- public class UserDAOImpl implements UserDAO {
- <br>
- <br> public void save() {
- <br> // TODO Auto-generated method stub
- <br> System.out.println("user saved");
- <br> }
- <br>
- <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代理。
- //相关配置,省略了一些不相关内容
- <br><bean id="userDAO" class="UserDAOImpl">
- <br><bean id="userDAOProxy" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
- <br> <property name="target">
- <br> <ref local="userDAO" />
- <br> </property>
- <br> <property name="optimize">
- <br> <value>true</value>
- <br> </property>
- <br> <property name="proxyTargetClass">
- <br> <value>true</value>
- <br> </property>
- <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。