【Spring】Spring和SpringMVC项目父子容器下Spring事务不生效问题 & SSM框架中,事务无法回滚的原因和解决

Spring + Spring MVC +Mybatis架构下,事务无法回滚的原因和解决

 

1.首先原理图

 

2. 父子容器下为什么事务不生效

①第一阶段 -- 容器初始化
一个项目中既有Spring,又有Spring MVC的情况下,默认web.xml配置如下。
Web容器(一般是Tomcate)启动,加载web.xml,就会开始上下文的加载。
1>先加载applicationContext.xml,完成Spring容器的初始化。
2>后加载springMVC-servlet.xml,完成Spring MVC容器的初始化。
3>由于一般情况下,两个xml文件中都会配置<context-conpent-scan>,即会开启注解的扫描。
4>因此,Spring容器和Spring MVC容器就都会扫描到 以@Conpent、@Controller、@Service、@Repository等作用在类上生效的注解管理的Bean




②第二阶段 -- 父子容器调用关系
再来理解父子容器的关系:
用一句话来理解,儿子可以继承并使用老子的资产,但儿子的财产一般不给老子用。
父子容器关系如此,Spring MVC作为子容器,是可以用到Spring父容器的Bean,反之父容器的Bean无法调用子容器的Bean.

5>Spring容器,父容器。
一般声明式事务的开启是在Spring容器中,
因此Spring容器里面扫描到的@Service声明的Bean,才有了底层AOP动态代理的支持,它就有事务的增强。
而Spring MVC子容器也有注解扫描能力,也扫描到了@Service标注的Bean,但是这个Bean只是一个原始的Bean,并没有事务增强的能力。

6>Spring MVC容器,子容器。
由于DispacherServlet作为前端控制器的存在,因此默认由它接收到来自前端的HTTP请求,并通过@RequestMapping等定位到url地址,进入由Spring MVC子容器扫描到的@Controller中。




③第三阶段 -- 那怎么会出现事务不生效呢?
7>Spring MVC容器的@Controller Bean,通过@Autowired或@Resource注解注入了@Service, 但它默认注入的,是Spring MVC容器扫描到的没有 事务增强能力的 普通Bean.
8>因此,这种情况下,就出现了事务不生效的问题。

 

 

<!-- 加载spring容器 -->
  <!-- 初始化加载application.xml的各种配置文件 -->
  <context-param>
    <param-name>contextConfigLocation</param-name>
    <param-value>classpath:spring/application-*.xml</param-value>
  </context-param>
  <listener>
    <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
  </listener>

  <!-- 配置springmvc前端控制器 -->
  <servlet>
    <servlet-name>taotao-manager</servlet-name>
    <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
    <!-- contextConfigLocation不是必须的, 如果不配置contextConfigLocation,
     springmvc的配置文件默认在:WEB-INF/servlet的name+"-servlet.xml" -->
    <init-param>
        <param-name>contextConfigLocation</param-name>
        <param-value>classpath:spring/springmvc.xml</param-value>
    </init-param>
    <load-on-startup>1</load-on-startup>
  </servlet>

 

3. 怎么解决父子容器下事务不生效

 

解决方案1:

1.在主容器中(applicationContext.xml),将Controller的注解排除掉:

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

2.而在springMVC配置文件中将Service注解给去掉:

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

3. 最终达到效果,就是@Controller Bean只在Spring MVC容器,@Service Bean只在Spring父容器

如此。通过@Controller Bean调用 @Service的Bean
就只能找到Spring父容器中的Bean,也就只能找到那个 被事务增强了的 @Service。
事务就可以生效了。

 

 

解决方案2:

取消 @Service 注解标注 Bean的方式,

使用XML配置的方式去 完成 Service类的Bean的配置

这样做,就导致SpringMVC子容器无法扫描到Service管理的Bean,只能将Service类的Bean交给Spring容器ServletContextListener去管理了。

这样,同样保证了 Service类的Bean只在Spring容器中,有了事务增强。

 

 

 

 

 

 

以下参考文章转载自:https://www.cnblogs.com/xiaojiesir/p/11058541.html

另参考文章:https://blog.csdn.net/weixin_34126557/article/details/92250009

=========================================================

原因:

  由ServletContextListener加载spring配置文件产生的是父容器,springMVC产生的是子容器,子容器对Controller进行扫描装配时装配了@Service注解的实例,而该实例理应由父容器进行初始化以保证事务的增强处理。所以此时得到的将是原样的Service(没有经过事务加强处理),故而没有事务处理能力。

第一种解决办法:

1.在主容器中(applicationContext.xml),将Controller的注解排除掉:

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

2.而在springMVC配置文件中将Service注解给去掉:

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

第二种解决办法:

  将service层改用xml配置,其实这样做也是变相的让springmvc无法扫描service,而只能依赖父窗口也就是ServletContextListener来进行初始化,这样同样被赋予了事务性。

 

Spring和SpringMVC的父子容器关系

1.讲问题之前要先明白一个关系

一般来说,我们在整合spring和SpringMVC这两个框架中,web.xml会这样写到:

复制代码
 1 <!-- 加载spring容器 -->
 2   <!-- 初始化加载application.xml的各种配置文件 -->
 3   <context-param>
 4     <param-name>contextConfigLocation</param-name>
 5     <param-value>classpath:spring/application-*.xml</param-value>
 6   </context-param>
 7   <listener>
 8     <listener-class>org.springframework.web.context.ContextLoaderListener</listener-class>
 9   </listener>
10  
11   <!-- 配置springmvc前端控制器 -->
12   <servlet>
13     <servlet-name>taotao-manager</servlet-name>
14     <servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
15     <!-- contextConfigLocation不是必须的, 如果不配置contextConfigLocation,
16      springmvc的配置文件默认在:WEB-INF/servlet的name+"-servlet.xml" -->
17     <init-param>
18         <param-name>contextConfigLocation</param-name>
19         <param-value>classpath:spring/springmvc.xml</param-value>
20     </init-param>
21     <load-on-startup>1</load-on-startup>
22   </servlet>
复制代码

 

首先配置的是Spring容器的初始化加载的application文件,然后是SpringMVC的前端控制器(DispatchServlet),当配置完DispatchServlet后会在Spring容器中创建一个新的容器。其实这是两个容器,Spring作为父容器,SpringMVC作为子容器。
让我们用图来看一下这个父子关系的原理:

 

平时我们在项目中注入关系是这样的顺序(结合图来说):在Service中注入Dao(初始化自动注入,利用@Autowired),接着在Controller里注入Service(初始化自动注入,利用@Autowired),看图,这就意味这作为SpringMVC的子容器是可以访问父容器Spring对象的。

那么问大家一个问题。要是反过来呢,你把Controller注入到Service中能行么?
肯定是不行的啊!(如图,这也说明了父容器是不能调用子容器对象的)

如果Dao,Serive,Controller要是都在Spring容器中,无疑上边的问题是肯定的,因为都是在一个bean里,一个容器中。

2.如果只用Spring容器,直接把所有层放入Spring容器中可不可以?

例如:一个项目中我总项目的名字叫com.shop,我们在配置applicationContext-service.xml中,包扫描代码如下:

 

复制代码
1 <?xml version="1.0" encoding="UTF-8"?>
2 <beans xmlns="http://www.springframework.org/schema/beans"
3 xmlns:context="http://www.springframework.org/schema/context" 
4 ...../ 此处省略>
5 
6 <!-- 扫描包Service实现类 -->
7 <context:component-scan base-package="com.shop.service"></context:component-scan>
8 </beans>
复制代码

上面所配置的是一个局部扫描,而不是全局扫描。接下来说原因:

这里就和上面讲到的父子容器有关系,假设我们做了全局扫描那么代码如下:

复制代码
1 <?xml version="1.0" encoding="UTF-8"?>
2 <beans xmlns="http://www.springframework.org/schema/beans"
3 xmlns:context="http://www.springframework.org/schema/context" 
4 ...../ 此处省略>
5 
6 <!-- 扫描包Service实现类 -->
7 <context:component-scan base-package="com.shop"></context:component-scan>
8 </beans>
复制代码

此时的Spring容器中就会扫描到@Controller,@Service,@Reposity,@Component,此时的图如下:

结合图去看,相当于他们都会放到大的容器中,而这时的SpringMVC容器中没有对象,没有对象就没有Controller,所以加载处理器,适配器的时候就会找不到映射对象,映射关系,因此在页面上就会出现404的错误。

3.如果不用Spring容器,直接把所有层放入SpringMVC容器中可不可以?

当然可以,如果没有Spring容器,我们是可以把所有层放入SpringMVC的。单独使用这个容器是完全可以的,而且是轻量级的。就是直接把所有的层次关系都放到了SpringMVC中,并没有用到Spring容器。

4.那么为什么我们在项目中还要联合用到Spring容器和SpringMVC容器?

答案是:Spring的扩展性,如果要是项目需要加入Struts等可以整合进来,便于扩展框架。如果要是为了快,为了方便开发,完全可以用SpringMVC框架。

    疑问一: 单例的bean在父子容器中存在一个实例还是两个实例?

    答:初始化两次,Spring 容器先初始化bean,MVC容器再初始化bean,所以应该是两个bean。

    疑问二:为啥不把所有bean 都在子容器中扫描?

    答: 网上很多文章说子容器不支持AOP,其实这是不对的。因为正常会有AOP的相关配置都在Spring容器中配置,如果都迁移到MVC配置文件,则所有bean都在子容器中,相当于只有一个容器了,所以也就实现了AOP。缺点是不利于扩展。

5.以上是传统型SSM(spring+springmvc+mybatis)配置时,父类容器负责扫描装配service层和dao层,子类容器负责扫描装配controller层,但是在轻量化的javaEE开发中,我们可以不用spring框架,只是利用spinrgmvc+mybatis即可

        具体操作:不使用listener监听器来加载spring的配置文件,只使用DispatcherServlet来加载spring的配置,不要父子上下文,只使用一个DispatcherServlet。将原本应该放在父容器中的数据源、服务层、DAO层、事务的Bean、以及子容器controller的Bean都放在子上下文容器中,这样也可以实现事务。

 

posted @ 2025-01-02 15:16  Angel挤一挤  阅读(11)  评论(0编辑  收藏  举报