springboot使用mybatis开启事务回滚
作者:岑惜 发布时间:2022-09-10 05:15:55
1.前言
以前没有使用mybatis,可以关闭自动提交,然后做sql操作,对操作进行catch捕获异常,
如果没有异常则commit 提交 ,有异常则 rollback 回滚,新增的数据则删除 ,修改的数据则修改回去,删除的则新增,
这就是事务操作。
事务有四大特性
(1)原子性:要么全部执行成功,要么不执行。
(2)一致性:事务执行的结果,必须使数据库从一个一致性状态变到另一个一致性状态。
(3)隔离性:并发操作同一个表时数据库会开启多个事务,多个事务之间相互隔离。
(4)持久性:当事务确认完成后,对数据的改变是永久性的。
那么mybatis怎么具体开启事务?
spring boot 开启其实很简单,使用注解开启即可,但是需要注意,需要触发非检查异常才会做事务回滚操作,【Exception 是检查异常】
但是如果使用try catch 捕获异常,也不会触发异常,因为异常被 吃下去了,做了服务降级操作,事务以为没有异常发生,因此不会触发回滚操作。
如果非要触发事务回滚,则需要在事务注解指定会触发事务回滚操作的异常类型,如果需要自定义抛出异常后反馈前端的数据,那么需要自定义异常,
自定义异常将会在下一随笔详细讲解。
经过测试总结:
(1)父级方法开启事务 @Transactional,父级发生异常,不仅父级会回滚,他调用的所有子方法都会回滚,也就是说,回滚事务父级可以影响所有子级.
(2)如果子级开了事务,父级没有开,发生异常,则仅仅让子级方法回滚,如果父级也开了事务,那么所有的子级将会和父级一起回滚。
2.操作
(1)提前配置好spring boot + mybatis
目录结构
红色箭头的文件是必要的,
(2)导入依赖包
完整源码
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 https://maven.apache.org/xsd/maven-4.0.0.xsd">
<modelVersion>4.0.0</modelVersion>
<parent>
<groupId>cen.cloud</groupId>
<artifactId>cen-mycloud</artifactId>
<version>0.0.1-SNAPSHOT</version>
<relativePath/> <!-- lookup parent from repository -->
</parent>
<groupId>com.example</groupId>
<artifactId>rabbitmq-producer-1004</artifactId>
<version>0.0.1-SNAPSHOT</version>
<name>rabbitmq-producer-1004</name>
<description>Demo project for Spring Boot</description>
<properties>
<java.version>1.8</java.version>
</properties>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
<scope>test</scope>
<exclusions>
<exclusion>
<groupId>org.junit.vintage</groupId>
<artifactId>junit-vintage-engine</artifactId>
</exclusion>
</exclusions>
</dependency>
<!--eureka 注册中心依赖包 -->
<dependency>
<groupId>org.springframework.cloud</groupId>
<artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
</dependency>
<!-- 消息中间件-->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-amqp</artifactId>
<version>2.1.6.RELEASE</version>
</dependency>
<!-- MySQL 依赖-->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<!-- <scope>runtime</scope>-->
<version>5.1.30</version>
</dependency>
<!--MySQL 数据源 依赖包-->
<dependency>
<groupId>com.alibaba</groupId>
<artifactId>druid-spring-boot-starter</artifactId>
<version>1.1.10</version>
</dependency>
<!-- mybatis依赖-->
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>2.1.2</version>
</dependency>
<!-- mybatis的逆向工程依赖包-->
<dependency>
<groupId>org.mybatis.generator</groupId>
<artifactId>mybatis-generator-core</artifactId>
<version>1.3.2</version>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
</plugin>
</plugins>
</build>
</project>
(3)启动类开启事务管理
(4)此时的数据库表信息
3.测试
(1)父级方法不开启事务,子级开启,让子级方法触发异常
启动后访问http://localhost:1004/sw
返回了500错误
报了个异常
查看数据库表信息
可见父级方法并没有回滚,子级方法事务回滚了
(2)恢复数据库表信息
父级方法不开启事务,子级开启,让子级方法catch捕获触发异常
父级方法不变,修改子级方法
启动后访问http://localhost:1004/sw
控制台打印
查看数据库
两次sql操作都执行了,子方法触发了异常,并没有做事务回滚操作,因为catch将服务降级了
那怎么办?
希望既可以做事务回滚操作,又能让前端获取指定的反馈信息怎么操作?
答案是手动抛出异常
throw new RuntimerException("这里写上你需要的骚话");
(3)恢复数据库表信息
父级方法不开启事务,子级开启,让子级方法catch捕获触发异常后,手动抛出异常
父级方法不变,修改子级方法
启动后访问http://localhost:1004/sw
控制台打印
查看数据库
可见,子级方法事务回滚了,但是父级没有,因为父级没有开启事务。
(4)如果使用 throw new Exception() 抛出异常则无法触发事务回滚
恢复数据库后,启动工程,访问http://localhost:1004/sw
查看数据库
可见,不能使用throw new Exception()
(5)恢复数据库
在事务注解指定抛出的异常则可以让检查性异常触发事务
父级方法不变,修改子级方法
启动工程,访问http://localhost:1004/sw
查看数据库
显然 ,子级方法做了事务回滚操作了,父级没影响
(6)好了这里开始需要修改父级啦,
在父级添加事务注解
子级方法不变
启动工程,访问http://localhost:1004/sw
查看数据库
显然,子级抛出异常,做了事务回滚操作,父级也做了事务回滚操作
(7)恢复数据库,删除子级方法事务注解,即关闭子级事务,父即开启事务
启动工程,访问http://localhost:1004/sw
查看数据库
显然,子级抛出异常,做了事务回滚操作,父级也做了事务回滚操作,即便子级没有开启事务,只有父级开启,
因此可见,只要父级开启了事务,不论是子级还是父级触发了非检查异常都会做事务回滚,如果是检查异常,则需要在事务注解指定异常类型。
(8)如果子级方法不触发异常,而是在父级触发,那么子级方法是否会回滚?
答案是会的
修改父级方法
修改子级方法
启动工程,访问http://localhost:1004/sw
查看数据库
显然,父级开启了事务且抛出异常,做了回滚操作,子级没有开启事务也没有抛出异常,仍然做了事务回滚操作
来源:https://www.cnblogs.com/c2g5201314/p/13163097.html


猜你喜欢
- 自定义类:using System;using System.Collections.Generic;using System.Linq;u
- Spring spring-context-indexer依赖<dependencies> <d
- 一个简单的照相功能,拍照之后在另一个activit中显示出拍照的图片。首先是布局文件:<?xml version="1.0&
- 一、前言程序界面上的按钮多种多样,常用的就这几种:普通按钮、图标按钮、文字按钮、图片文字混合按钮。本文章记录了不同样式类型的按钮实现方法。下
- java分别解析下面两个json字符串package jansonDemo;import com.alibaba.fastjson.JSON
- 在面向对象设计原则中,要求"要依赖于抽象,不要依赖于具体", 这句话有很多人搞不懂。在这里谈谈我自己的理解。首先看看以下
- 引言内存管理一直是JAVA语言自豪与骄傲的资本,它让JAVA程序员基本上可以彻底忽略与内存管理相关的细节,只专注于业务逻辑。不过世界上不存在
- 大概理解查了一个小时的资料:async和await发现这个大神的解释一针见血,深得我心!以最简单的例子,解释了async和await。妙~~
- 单点登录三种方式单点登录的三种实现方式:分别为session广播机制;cookie+redis;tokensession广播机制指在一个集群
- 一、数组的基本用法1.什么是数组数组:存储一组相同数据类型的数据的集合。2.定义数组 int[] :int类型数组 do
- Mybatis-Plus是一个优秀的Mybatis增强工具,目前更新到3.1.1。Mybatis-Plus原生提供了很多单表操作的方法,极大
- 本文实例为大家分享了Android实现APP秒表功能的具体代码,供大家参考,具体内容如下这几天一直在看安卓,也正好赶上老师布置的作业,所以就
- 本文实例为大家分享了Intent实现页面跳转的两种的方法,供大家参考,具体内容如下下图中两个不同的方法就是两种页面之间跳转的情况1).跳转不
- IO操作字节流java.io.InputStream 输入流,主要是用来读取文件内容的。java.io.OutputStream 输出流,主
- Android ListView的优化,在做Android项目的时候,在用到ListView 界面及数据显示,这个时候如果资源过大,对项目来
- 1.线程状态(生命周期)一个线程在给定的时间点只能处于一种状态。线程可以有如下6 种状态:New (新创建):未启动的线程;Runnable
- 个人详情页滑动到顶部最近产品提了个新需求,需要实现点击App内的某个按钮跳转到个人详情页并且滑动到顶部,个人详情页的页面交互稍微复杂,技术角
- TCP实现TCP协议需要在双方之间建立连接,通过输入输出流来进行数据的交换,建立需要通过三次握手,断开需要四次挥手,保证了数据的完整性,但传
- 匿名类是不能有名称的类,所以没办法引用它们。必须在创建时,作为new语句的一部分来声明它们。这就要采用另一种形式的new语句,如下所示: n
- 当我们的idea无法自动下在所需的Maven依赖时,我们可以到Maven的远程仓库中下载所需要的jar包,然后添加到我们的本地仓库中。1.首