一、知识点及问题
后端框架:
Spring 、Spring mvc 、mybatis
业务需求:
客户端先从服务端获取用户大量信息到客户端,编辑完成之后统一Post至服务端,对于数据的修改要么全成功,要么全失败,所以需要使用事务支持。
问题:
配置Spring声明式事务,执行中出现异常未回滚.从网上查询得到一开始是自己的配置出了问题,由于配置文件的加载顺序决定了容器的加载顺序导致Spring事务没有起作用。详情如下:
由于采用的是SpringMVC、 MyBatis,故统一采用了标注来声明Service、Controller
由于服务器启动时的加载配置文件的顺序为web.xml—root-context.xml(Spring的配置文件)—servlet-context.xml(SpringMVC的配置文件),由于root-context.xml配置文件中Controller会先进行扫描装配,但是此时service还没有进行事务增强处理,得到的将是原样的Service(没有经过事务加强处理,故而没有事务处理能力),所以我们必须在root-context.xml中不扫描Controller
上面的问题解决后还是没有回滚,后来了解到,Spring 只会在程序执行中出现unchecked(RuntimeException)的异常时才会触发回滚。由于是与客户端直接交互的Server所以要将每一个处理结果以 errorcode 错误码和msg 错误信息的形式反馈给客户端所以显式捕捉了所有的异常,并将信息以Json数据格式发送给客户端这才导致了出现异常时事务没有回滚。
因为要给客户端最真实、准确的错误信息反馈又不得不捕捉可能发生的异常又陷入了沉思.当然,问题总是有解决的方式的,哪怕是绕着走。之后从查询资料得到,捕捉可以,但是捕捉之后主动抛出还是会引发事务回滚的!(喜)然后就想到在主动 throw new RuntimeException(“反馈给客户端的信息”);将要反馈给客户端的具体错误信息包装到异常信息中,发生异常时在Controller层catch异常,将信息返回至客户端。
(mysql 表的engine为InnoDB–支持事务回滚,默认为MyISAM–效率高)
到此,问题解决。
二、案例声明
岗位: Java服务端
工作内容: 接收来自客户端的请求(android,androidtv,ios,pc ..),对客户端请求数据做合法性校验,并与其他服务端交互获取客户端所需数据。
三、代码及配置
1.web.xml配置
<?xml version="1.0" encoding="UTF-8"?>
<listener>
<listener-class>
org.springframework.web.context.ContextLoaderListener
</listener-class>
</listener>
<context-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:applicationContext.xml</param-value>
</context-param>
<servlet>
<servlet-name>spring-mvc</servlet-name>
<servlet-class>org.springframework.web.servlet.DispatcherServlet</servlet-class>
<init-param>
<param-name>contextConfigLocation</param-name>
<param-value>classpath:springmvc-servlet.xml</param-value>
</init-param>
<load-on-startup>1</load-on-startup>
</servlet>
2. Spring-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:p="http://www.springframework.org/schema/p"
xmlns:co