spring 事务管理配置无效--分析与解决方案

配置如下:

<tx:advice id="txAdvice" transaction-manager="txManager">
    <tx:attributes>
        <tx:method name="get*" read-only="true"/>
        <tx:method name="query*" read-only="true"/>
        <tx:method name="*"/>
    </tx:attributes>
</tx:advice>
<!-- AOP切入点设置 -->
<aop:config>
    <aop:pointcut id="dbService" expression="execution(* com.qyd.sms.service.api.student.service.StudentService.*(..))" />
    <aop:advisor advice-ref="txAdvice" pointcut-ref="dbService" />
</aop:config>

事务管理表现为无效,同时还有一个非常奇怪的现象:如果将com.qyd.sms.service.api.student.service.StudentService这个类,以<bean/>的形式显式的配置在配置文件中,事务管理就可以生效

不得其解,很幸运在网上找到了这个:spring+springMVC,声明式事务失效,原因以及解决办法,结合目前的情况(显式配置生效,注解不生效),以及这篇文章的内容,无效的原因应该是StudentService没有被spring加载,而是被springMVC加载,而srpingMVC加载时没有处理事务相关的部分,所以事务配置无效

有起色,去着重看一下项目的spring、springMVC的启动方式,

<listener>
    <listener-class>
        org.springframework.web.context.ContextLoaderListener
    </listener-class>
</listener>
<context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>classpath*:/META-INF/spring/spring-beans.xml</param-value>
</context-param>

<servlet>
    <servlet-name>qyd-sms-servlet</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <init-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath*:/META-INF/spring/spring-web.xml</param-value>
    </init-param>
    <load-on-startup>1</load-on-startup>
</servlet>

<servlet-mapping>
    <servlet-name>qyd-sms-servlet</servlet-name>
    <url-pattern>/*</url-pattern> </servlet-mapping>

spring是通过ContextLoaderListener启动,springMVC通过DispatcherServlet启动,经过验证,spring 先于 springMVC 启动(暂时将ContextLoaderListener启动的部分称为spring,表示spring核心的部分,而springMVC则是处理请求的部分)

当前配置:spring启动配置了事务管理,没有配置自动扫描,springMVC配置了全局的自动扫描

事务无效的原因分析:
spring启动时能够拿到事务管理信息,但是由于没有配置扫描配置,所以看不到通过注解注册的StudentService,也就无法进行StudentService的事务相关的处理

等到springMVC启动时,可以注册StudentService,但是没有事务管理相关的配置,也就不会做事务相关的处理(tip:我不会尝试在springMVC中加入事务管理配置,去试一下是否事务能生效,因为即使能生效,也不会采用这种做法,理解即可)

所以 只要在spring启动时看到注册StudentService,就能够进行事务相关的处理,事务就能够生效,这也解释了为什么显式配置一个bean就可以生效,通常我们不会去显式配置每一个service,所以spring配置扫描即可解决该问题

我的解决方案:

springMVC:

<mvc:annotation-driven/>
<context:component-scan base-package="com.qyd.sms.controller" />

spring:

<context:component-scan base-package="com.qyd.sms" >
    <context:exclude-filter type="annotation" 
    expression="org.springframework.stereotype.Controller"/>     
</context:component-scan>

配置文件如上,原因下面有说明。

说明:
经过测试,springMVC Controller的注册是不能放在spring启动时进行的,因为springMVC拿到注册的Controller信息后进行了的进一步处理,用于接收来访的请求

我更倾向于,spring,springMVC使用同一个bean容器,springMVC完全是基于spring来处理依赖注入的关系的,只不过springMVC做了额外的事

既然二者功能不同,对于配置文件的话,我比较倾向于根据功能完全分开,springMVC只负责了Controller层的注册的扫描,spring则负责其它所有。

你可能感兴趣的:(spring)