springmvc mybatis 声明式事务管理回滚失效,(checked回滚)捕捉异常,传输错误信息

springmvc mybatis 声明式事务管理回滚失效,(checked回滚)捕捉异常,传输错误信息_第1张图片

一、知识点及问题

后端框架:
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配置




        org.springframework.web.context.ContextLoaderListener



contextConfigLocation
classpath:applicationContext.xml
  
  
spring-mvc
org.springframework.web.servlet.DispatcherServlet

  contextConfigLocation
  classpath:springmvc-servlet.xml

1
  

2. Spring-servlet.xml配置




    
        
        
    
  

3. mybatis-config.xml配置




    
        
        
        
        
        
    
    
    
        
            
        
    

4. applicationContext.xml配置




    
    
        
    
    
    
        

        

        

        

        

        

        

        

        

        

        

        

        

        



        

        

            5

        

        

        

            0

        

        

        

            60

        

        

        

            30

        

        

        

            true

        

        

        

            true

        

    

    

    
        
    

    

        

        

        

    

    

    

        

    
    
    
    
        
    
    
    
        
        
    
    
    
        
            
            
        
    

5. 代码示例

Service业务逻辑处理层

try {
    log.info("check phone is exist before ..");
    int count = tMailingMapper.insert(record);
        if (count > 0) {
                log.info("添加 " + accountid + " 的好友"
                        + account.getAccountid() + "  "
                        + phoneVo.getRemark() + "  成功");
        } else {
                log.info("添加 " + accountid + " 的好友"
                        + account.getAccountid() + "  "
                        + phoneVo.getRemark() + "  失败");
        }

} catch (Exception e) {
    log.error("添加联系人出现了异常 " + e.getMessage());
    resJson.put("errorcode", "20022");
    resJson.put("msg", "同步信息异常,请稍后重试");
    throw new RuntimeException(resJson.toString());
}

Controller 控制层

@RequestMapping("/update/userinfo")
    public String updateUserInfo(HttpServletRequest request, HttpServletResponse response) {
        log.info("update userInfo start ..");
        String resText=null;
        try {
            resText = userService.updateUserInfo(request);
        } catch (Exception e) {
            log.error("更新信息失败,事务已回滚...",e);
            resText=e.getMessage();
        }
        <-- 将操作结果返回给client-->
        HttpsUtil.sendAppMessage(resText, response);
        return null;
    }

本人初入Java,小白一枚,有不到之处还请见谅。欢迎大神指点!

2016-03-19 10:58:58

参考文章:1. http://sence-qi.iteye.com/blog/1328902/
2.http://blog.sina.com.cn/s/blog_89ca421401016bmg.html

你可能感兴趣的:(JavaWeb,Java,Spring,学习感悟)