zoukankan      html  css  js  c++  java
  • spring之事务回滚技巧

    1、遇到的问题

    当我们一个方法里面有多个数据库保存操作的时候,中间的数据库操作发生的错误。伪代码如下:

    public method() {
       Dao1.save(Person1);
       Dao1.save(Person2);
       Dao1.save(Person2);
       //假如这句发生了错误,前面的两个对象会被保存到数据库中
       Dao1.save(Person2);
    }

    期待的情况:发生错误之前的所有数据库保存操作都回滚,即不保存

    正常情况:前面的数据库操作会被执行,而发生数据库操作错误开始及之后的所有的数据保存操作都将失败。这样子应该都不是我们要的结果吧。

    当遇到这种情况,我们就可以使用Spring的事务解决这个问题。

    2、异常的一些基本知识

    1) 异常的架构

    Throwable为基类,Error和Exception继承Throwable,RuntimeException和IOException等继承Exception。

    Error和RuntimeException及其子类成为未检查异常(unchecked),其它异常成为已检查异常(checked)。 

    2)Error异常

    Error表示程序在运行期间出现了十分严重、不可恢复的错误,在这种情况下应用程序只能中止运行,例如JAVA 虚拟机出现错误。Error是一种unchecked Exception,编译器不会检查Error是否被处理,在程序中不用捕获Error类型的异常。一般情况下,在程序中也不应该抛出Error类型的异常。

    3)RuntimeException异常

    Exception异常包括RuntimeException异常和其他非RuntimeException的异常。

      

    RuntimeException 是一种Unchecked Exception,即表示编译器不会检查程序是否对RuntimeException作了处理,在程序中不必捕获RuntimException类型的异常,也不必在方法体声明抛出RuntimeException类。

    RuntimeException发生的时候,表示程序中出现了编程错误,所以应该找出错误修改程序,而不是去捕获RuntimeException。

    4)Checked Exception异常

    Checked Exception异常,这也是在编程中使用最多的Exception,所有继承自Exception并且不是RuntimeException的异常都是checked Exception,上图中的IOException和ClassNotFoundException。

    JAVA 语言规定必须对checked Exception作处理,编译器会对此作检查,要么在方法体中声明抛出checked Exception,要么使用catch语句捕获checked Exception进行处理,不然不能通过编译。

    3、实例

    这里使用的事务配置如下:


       bean id='transactionManager' class='org.springframework.orm.jpa.JpaTransactionManager'>
           property name='entityManagerFactory' ref='entityManagerFactory'/>
       bean>
       
       
       tx:annotation-driven transaction-manager='transactionManager' proxy-target-class='true' />

    在spring的配置文件中,如果数据源的defaultAutoCommit设置为True了,那么方法中如果自己捕获了异常,事务是不会回滚的,如果没有自己捕获异常则事务会回滚,如下例

    比如配置文件里有这么条记录

    'dataSource' class='com.alibaba.druid.pool.DruidDataSource'>

    'xxx' value='xxx'/>

    'xxx' value='xxx'/>

    ....
    'defaultAutoCommit' value='true' />

    可能你会发现你并没有配置这个参数,是不是他就不会自动提交呢?答案是不是的,我这里是使用了com.alibaba.druid.pool.DruidDataSource作为数据库连接池,默认的defaultAutoCommit就是true,可以看下面的源码

     

    那么现在有两个情况

      

    情况1:如果没有在程序中手动捕获异常

    @Transactional(rollbackOn = { Exception.class })  
    public void test() throws Exception {  
        doDbStuff1();  
        doDbStuff2();
    //假如这个操作数据库的方法会抛出异常,
    //现在方法doDbStuff1()对数据库的操作   会回滚。  
    }

    情况2:如果在程序中自己捕获了异常

    @Transactional(rollbackOn = { Exception.class })  
    public void test() {  
        try {  
           doDbStuff1();  
           doDbStuff2();
    //假如这个操作数据库的方法会抛出异常,
    //现在方法doDbStuff1()对数据库的操作  不会回滚。  
        } catch (Exception e) {  
              e.printStackTrace();    
        }  
    }

    现在如果我们需要手动捕获异常,并且也希望抛异常的时候能回滚肿么办呢?

      

    下面这样写就好了,手动回滚事务:

    @Transactional(rollbackOn = { Exception.class })  
    public void test() {  
       try {  
          doDbStuff1();  
          doDbStuff2();  
       } catch (Exception e) {  
            e.printStackTrace();    
            TransactionAspectSupport.currentTransactionStatus()
            .setRollbackOnly();
        //就是这一句了,加上之后,如果doDbStuff2()抛了异常,
        //doDbStuff1()是会回滚的  
       }  
    }
  • 相关阅读:
    【leetcode】538/1038: 把二叉搜索树转化为累加树
    k8s-nginx二进制报Illegal instruction (core dumped)
    k8s-记一次安全软件导致镜像加载失败
    Ubuntu1804下k8s-CoreDNS占CPU高问题排查
    Ubuntu 18.04 永久修改DNS的方法
    NLP资源
    《转载》14种文本分类中的常用算法
    PyCharm 使用技巧
    python模块包调用问题
    强化学习(8)------动态规划(通俗解释)
  • 原文地址:https://www.cnblogs.com/leeego-123/p/12132884.html
Copyright © 2011-2022 走看看