|
精华帖 (0) :: 良好帖 (0) :: 新手帖 (0) :: 隐藏帖 (0)
|
|
|---|---|
| 作者 | 正文 |
|
时间:2008-04-14 关键字: aop
测试代码 ApplicationContext ctx = 上面这种情况下程序可以正常运行,但是如果UserDAOImpl实现了一个接口,其他不变 public class UserDAOImpl implements UserDAO {
这种情况下,程序将不能正常运行,会抛出java.lang.ClassCastException异常 理解上面这种情况产生的原因需要了解Spring AOP的实现原理。
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代理。
使用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。
声明:JavaEye文章版权属于作者,受法律保护。没有作者书面许可不得转载。
|
|
| 返回顶楼 | |
|
时间:2008-04-14
UserDAOImpl userDAOImpl = (UserDAOImpl)ctx.getBean("userDAOProxy");
类型转换的时候肯定会有错. 你用这个得到的实际上是一个FactoryBean .也就是具有创建bean能力的bean.试问一个Factory 怎么可以cast成为一个UserDao的实现类呢? spring的getBean方法可以得到两种类型的.一种就是代理FactoryBean 也就是可以产生出其他对象的. 另外一种就是不具有这种创建其他对象的能力的. |
|
| 返回顶楼 | |
|
时间:2008-04-14
理解上面这种情况产生的原因需要了解Spring AOP的实现原理。 //org.springframework.aop.framework.DefaultAopProxyFactory
advisedSupport.isOptimize()与advisedSupport.isProxyTargetClass()默认返回都是false,所以在默认情况下目标对象有没有实现接口决定着Spring采取的策略,当然可以设置advisedSupport.isOptimize()或者advisedSupport.isProxyTargetClass()返回为true,这样无论目标对象有没有实现接口Spring都会选择使用CGLIB代理。所以在默认情况下,如果一个目标对象如果实现了接口Spring则会选择JDK动态代理策略动态的创建一个接口实现类(动态代理类)来代理目标对象,可以通俗的理解这个动态代理类是目标对象的另外一个版本,所以这两者之间在强制转换的时候会抛出j ava.lang.ClassCastException。而所以在默认情况下,如果目标对象没有实现任何接口,Spring会选择CGLIB代理, 其生成的动态代理对象是目标类的子类。
|
|
| 返回顶楼 | |
|
时间:2008-04-14
lsk 写道 UserDAOImpl userDAOImpl = (UserDAOImpl)ctx.getBean("userDAOProxy");
类型转换的时候肯定会有错. 你用这个得到的实际上是一个FactoryBean .也就是具有创建bean能力的bean.试问一个Factory 怎么可以cast成为一个UserDao的实现类呢? spring的getBean方法可以得到两种类型的.一种就是代理FactoryBean 也就是可以产生出其他对象的. 另外一种就是不具有这种创建其他对象的能力的. 两种情况返回的都是动态代理类,只是与UserDAOImpl关系有所不同 在默认情况下UserDAOImpl userDAOImpl = (UserDAOImpl)ctx.getBean("userDAOProxy");类型转换有没有错与UserDAOImpl有没有实现接口有关 |
|
| 返回顶楼 | |
|
时间:2008-04-14
bleakoasis回答很精彩~今天有学到东西了。呵呵~
|
|
| 返回顶楼 | |
|
时间:2008-04-14
应该这样使用
UserDAO userDAOImpl = (UserDAO)ctx.getBean("userDAOProxy"); userDAOImpl.save(); |
|
| 返回顶楼 | |
浏览 622 次




