在近期一个项目中用了Spring MVC作为控制层框架,但却出现了一个让人非常费解的问题:事务控制。
Spring MVC的配置文件名称为:springMVC-servlet.xml,内容例如以下:
<?
xml version="1.0" encoding="UTF-8"?
>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:mvc="http://www.springframework.org/schema/mvc"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="
">
<!-- 基于注解的配置 -->
<context:component-scan base-package="com.xtayfjpk.esb.console"/>
<!-- 启动Spring MVC的注解功能。完毕请求和注解POJO的映射 -->
<bean />
<bean/>
<bean>
<property value="/app/"/>
<property value=".jsp"/>
<property value="org.springframework.web.servlet.view.JstlView"/>
</bean>
</beans>
Spring配置文件名称为:applicationContext.xml,内容例如以下:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:util="http://www.springframework.org/schema/util"
xsi:schemaLocation="http://www.springframework.org/schema/beans
">
<!-- 数据源 -->
<bean>
<property value="com.mysql.jdbc.Driver"/>
<property value="jdbc:mysql://localhost:3306/esb_center?useUnicode=true&charset=UTF-8"/>
<property value="root"/>
<property value="xtayfjpk"/>
</bean>
<bean>
<property ref="dataSource"/>
<property value="classpath:sqlMapConfig.xml"/>
</bean>
<!-- 配置声明式的事务管理(採用基于注解的方式) -->
<bean>
<property ref="dataSource"/>
</bean>
<!-- 启用切面编程 -->
<aop:aspectj-autoproxy />
<!-- 启用注解 -->
<context:annotation-config/>
<!-- 注解驱动的事务管理器 -->
<tx:annotation-driven transaction-manager="txManager"/>
</beans>
我想大家对于这两个配置文件都非常熟悉。但我遇到的问题是事务管理器并没有起作用,当发生异常时没有按预期进行事务回滚。
非常纳闷,检查了好几遍配置都没有问题,但问题出现了,那肯定是自已哪里出了问题。
后来手动用代码实例化出容器再调用业务层方法时,事务管理器却起作用了。异常时事务正常回滚。这就奇怪了,为什么手动调用业务层方法时。事务管理器起作用,而通过页面调用时去不起作用了呢,后经一步步排查,对照,当把
<tx:annotation-driven transaction-manager="txManager"/>
这一配置从applicationContext.xml移至springMVC-servlet.xml文件时,通过页面调用事务管理器起作用了。而手动实例化容器调用时。事务管理器不起作用了,最后在这两个配置文件里都加上
<tx:annotation-driven transaction-manager="txManager"/>时,两种方式事务管理器都正常工作了。
由于本人暂对Spring事务管理器的具体工作机制不是非常了解,但我自己推測是不同容器之间在事务管理上有各自的一套机制(但可能使用同一个事务管理器对象),是不能共享的,即使两个容器之间存在着继承关系。所以在每个容器的配置文件里都要加上<tx:annotation-driven transaction-manager="txManager"/>这一配置(仅限于注解驱动的事务管理器)