关闭

Hibernate + Spring + Struts中的一个常见错误

321人阅读 评论(0) 收藏 举报
        使用Spring中的AOP技术代理了Service(实现声明式事务管理),然后将Service通过IoC注入给Action.未加入Struts时一切正常,但是加入Struts后出错,提示信息大意为无法将被代理的Service对象注入给Action,异常为ClassCastException.

    【分析原因】

    ClassCastException表明这是一个类型不匹配错误,再看详细的出错信息:"Can't convert $Proxy1 to com.xaccp.XxxService"(大体是这个意思,原文记不清了),说明在Action中需要的类型为com.xaccp.XxxService,而实际传入的是$Proxy1。

    接下来应该想到Service对象是被Spring代理的,Spring中有两种实现代理的方法:JDK提供的动态代理CGLIB

    CGLIB的实现原理是动态生成目标类(Target)的子类,如果一个类采用CGLIB代理的,生成的子类类名一般为com.xaccp.XxxService$$EnhanceByCGLIB$xxxx的格式,并且根据父子类转换的规则,在要求目标类的地方,如果传入生成的代理类不会发生ClassCastException异常。

    而动态代理的实现原理是动态生成一个与目标类显示相同接口的类,生成类类名就是$Proxy1这样的格式,在要求目标类的地方,如果传入生成的代理类就会发生上面的错误。

    再看学生写的Service类,果然实现了一个接口,那么默认的情况下,如果类存在接口,Spring就会采用动态代理来实现AOP,于是发生了上面的错误。

【解决方案】

    分析清楚原因以后解决方案就出来了:

    最简单的一种方案就是去掉Service实现的接口,那么Spring会采用CGLIB来实现AOP。但是这个方案显然破坏了代码的结构

    最应该采取的一种方案是在Action中不要直接将属性声明为Service类,而是声明为其接口,这样Spring就可以将实现了同样接口的$Proxy1代理类注入进来。并且体现了面向接口编程的原则。

    还有一种方案就是在设置代理时指明proxyTargetClass属性为true,强制Spring采用CGLIB进行代理,如下所示:

<bean id="service" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean">
 <property name="target">
  <ref bean="dummyService"/>
 </property>
  <property name="proxyTargetClass">
   <value>true</value>
  </property>
 <property name="transactionManager">
  <ref bean="tm"/>
 </property>
 <property name="transactionAttributes">
  <props> 
   <prop key="*">PROPAGATION_REQUIRED</prop>
  </props>
 </property>
</bean>

 

0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:2971次
    • 积分:90
    • 等级:
    • 排名:千里之外
    • 原创:6篇
    • 转载:0篇
    • 译文:0篇
    • 评论:1条
    文章存档