springMVC重复扫描bean导致声明式事务失效
spring和springMVC进行整合时,经常出现的问题是spring配置的声明式事务失效。
本文主要解决此问题。
1 配置文件
1.1 加载spring容器配置
通过配置listener
,初始化spring
。web.xml中,增加如下配置:
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>
classpath:conf/spring.xml
</param-value>
</context-param>
<listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
</listener>
其中,spring的配置文件是spring.xml
1.2 加载springMvc容器配置
通过配置servlet
,初始化springMVC
。web.xml中,增加如下配置(如同时初始化spring,需配置在spring的listener
的后面
):
<servlet>
<servlet-name>spring</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:conf/spring-mvc.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
<servlet-mapping>
<servlet-name>spring</servlet-name>
<url-pattern>/</url-pattern>
</servlet-mapping>
其中,springMvc的配置文件是spring-mvc.xml
1.3 spring声明式事务配置
spring配置文件中,对数据库配置声明式事务:
<!--事务管理器-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
<!--事务增强-->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="save*" propagation="REQUIRED" rollback-for="Exception"/>
<tx:method name="del*" propagation="REQUIRED" rollback-for="Exception"/>
<tx:method name="update*" propagation="REQUIRED" rollback-for="Exception"/>
<tx:method name="find*" propagation="REQUIRED" read-only="true"/>
<tx:method name="get*" propagation="REQUIRED" read-only="true"/>
</tx:attributes>
</tx:advice>
<!--定义切面-->
<aop:config>
<!-- 定义切入点 -->
<aop:pointcut id="txPointcut" expression="execution(* com.scaf.service..*.*(..))" />
<!-- 将事务增强与切入点结合 -->
<aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/>
</aop:config>
2 声明式事务失效
通过配置文件的加载配置,可以看到spring和springMvc的配置文件加载是分开进行
的,spring的context是父容器
,springMvc的context是子容器
。
2.1 事务失效的原因
springMvc进行扫描装配时,对controller依赖的service进行了装配
,得到了未通过spring进行事务加强的service的bean
(通过父容器spring扫描装配的service才支持事务),故事务失效。
详细解释如下:
(1)spring的context中对通过切面事务进行了加强处理,springMvc命名空间并未对事务进行加强处理。
(2)springMVC在装配bean时,对@Service的bean进行加载。
2.2 解决方案
(1)spring
进行bean的装配时,不装配@Controller
spring.xml中配置包扫描:
<!-- 通过扫描器实现bean的自动载入(只扫描非Controller注解,防止ContextLoaderListener扫描的事务失效) -->
<context:component-scan base-package="com.sa.dev.scaf">
<context:exclude-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>
(2)springMVC
进行bean的装配时,只装配@Controller
spring-mvc.xml中配置包扫描:
<!-- 通过扫描器实现bean的自动载入(只扫描Controller注解,防止ContextLoaderListener扫描的事务失效)
use-default-filters="false"需和include-filter配合使用,否则默认仍然会扫描所有注解-->
<context:component-scan base-package="com.sa.dev.scaf" use-default-filters="false">
<context:include-filter type="annotation" expression="org.springframework.stereotype.Controller"/>
</context:component-scan>
3 其他
3.1 spring、springMVC的父子容器
3.1.1 父子容器的关系
spring和springMvc进行整合,一般会分开配置xml文件
。spring
通过listener
加载配置文件,springMvc
通过servlet
加载配置文件。
spring
的容器会优先进行初始化
,spring容器初始化完毕后,初始化springMvc容器。spring容器将被设置为springMvc的父容器
。
容器 | 类型 | 加载bean |
---|---|---|
spring | 父容器 | Service、Dao... |
springMvc | 子容器 | Controller、HandleMapping、ViewResolver... |
3.1.2 为什么要有父子容器
分层,解耦。
将spring层与springMVC层分开。
3.1.3 注意事项
(1)springMvc在查找bean时,如当前容器不存在,将尝试去父容器查找
(2)spring、springMVC加载bean时,会重复加载相同的bean
(3)可参照本文解决事务失效问题的方式,将spring、springMVC分开加载各自需要的bean