spring boot:方法中使用try...catch导致@Transactional事务无效的解决(spring boot 2.3.4)
一,方法中使用try...catch导致@Transactional事务无效的解决方法
1,问题的描述:
如果一个方法添加了@Transactional注解声明事务,
而方法内又使用了try catch 捕捉异常,
则方法内的异常捕捉会覆盖事务对异常的判断,
从而异致事务失效而不回滚
2, 如何解决?
第一个方法:给@Transactional注解增加:rollbackFor后并手动抛出指定的异常
第二个方法:在捕捉到异常后手动rollback
说明:刘宏缔的架构森林是一个专注架构的博客,地址:https://www.cnblogs.com/architectforest
对应的源码可以访问这里获取: https://github.com/liuhongdi/
说明:作者:刘宏缔 邮箱: 371125307@qq.com
二,演示项目的相关信息
1,项目地址:
https://github.com/liuhongdi/transactional
2,功能说明:
演示了事务方法中捕捉异常时如何使事务回滚
3,项目结构:如图:
三,配置文件说明
1,pom.xml
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency> <!--mybatis begin-->
<dependency>
<groupId>org.mybatis.spring.boot</groupId>
<artifactId>mybatis-spring-boot-starter</artifactId>
<version>1.3.2</version>
</dependency> <!--mysql begin-->
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<scope>runtime</scope>
</dependency>
2,application.properties
#mysql
spring.datasource.url=jdbc:mysql://localhost:3306/store?characterEncoding=utf8&useSSL=false
spring.datasource.username=root
spring.datasource.password=password
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver #mybatis
mybatis.mapper-locations=classpath:/mapper/*Mapper.xml
mybatis.type-aliases-package=com.example.demo.mapper
3,创建数据表goods的sql
CREATE TABLE `goods` (
`goodsId` bigint(11) unsigned NOT NULL AUTO_INCREMENT COMMENT 'id',
`goodsName` varchar(500) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NOT NULL DEFAULT '' COMMENT 'name',
`subject` varchar(200) NOT NULL DEFAULT '' COMMENT '标题',
`price` decimal(15,2) NOT NULL DEFAULT '0.00' COMMENT '价格',
`stock` int(11) NOT NULL DEFAULT '0' COMMENT 'stock',
PRIMARY KEY (`goodsId`),
UNIQUE KEY `goodsName` (`goodsName`)
) ENGINE=InnoDB AUTO_INCREMENT=0 DEFAULT CHARSET=utf8mb4 COLLATE=utf8mb4_0900_ai_ci COMMENT='商品表'
注意goodsName字段上有一个唯一的索引,
后面我们会利用它来引发一个duplicate entry 异常
插入一条数据:
INSERT INTO `goods` (`goodsId`, `goodsName`, `subject`, `price`, `stock`) VALUES
(3, '100分电动牙刷', '好用到让你爱上刷牙', '59.00', 100);
四,java代码说明
1,LockController.java
@RestController
@RequestMapping("/lock")
public class LockController { @Resource
OrderService orderService; //购买商品,方法内没有捕捉异常
@GetMapping("/lock")
@ResponseBody
public String buyLock() {
try {
int goodsId = 3;
orderService.decrementProductStoreLock(goodsId,1);
return "success";
} catch (Exception e){
System.out.println("捕捉到了异常 in controller");
e.printStackTrace();
String errMsg = e.getMessage();
return errMsg;
}
} //购买商品,方法内捕捉异常
@GetMapping("/lockcatch")
@ResponseBody
public String buyLockCatch() {
try {
int goodsId = 3;
boolean isDecre = orderService.decrementProductStoreLockWithCatch(goodsId,1);
if (isDecre) {
return "success";
} else {
return "false";
}
} catch (Exception e){
System.out.println("捕捉到了异常 in controller");
e.printStackTrace();
String errMsg = e.getMessage();
return errMsg;
}
} //购买商品,方法内捕捉异常
@GetMapping("/lockcatch1")
@ResponseBody
public String buyLockCatch1() {
try {
int goodsId = 3;
boolean isDecre = orderService.decrementProductStoreLockWithCatch1(goodsId,1);
if (isDecre) {
return "success";
} else {
return "false";
}
} catch (Exception e){
System.out.println("捕捉到了异常 in controller");
e.printStackTrace();
String errMsg = e.getMessage();
return errMsg;
}
} //购买商品,方法内捕捉异常
@GetMapping("/lockcatch2")
@ResponseBody
public String buyLockCatch2() {
try {
int goodsId = 3;
boolean isDecre = orderService.decrementProductStoreLockWithCatch2(goodsId,1);
if (isDecre) {
return "success";
} else {
return "false";
}
} catch (Exception e){
System.out.println("捕捉到了异常 in controller");
e.printStackTrace();
String errMsg = e.getMessage();
return errMsg;
}
} }
2,GoodsMapper.java
@Repository
@Mapper
public interface GoodsMapper {
Goods selectOneGoods(int goods_id);
int updateOneGoodsStock(Goods goodsOne);
int insertOneGoods(Goods goodsOne);
}
3,Goods.java
public class Goods {
//商品id
private int goodsId;
public int getGoodsId() {
return this.goodsId;
}
public void setGoodsId(int goodsId) {
this.goodsId = goodsId;
} //商品名字
private String goodsName;
public String getGoodsName() {
return this.goodsName;
}
public void setGoodsName(String goodsName) {
this.goodsName = goodsName;
} //商品库存数
private int stock;
public int getStock() {
return this.stock;
}
public void setStock(int stock) {
this.stock = stock;
}
}
4,OrderService.java
public interface OrderService {
public boolean decrementProductStoreLock(int goodsId, int buyNum);
public boolean decrementProductStoreLockWithCatch(int goodsId, int buyNum);
public boolean decrementProductStoreLockWithCatch1(int goodsId, int buyNum);
public boolean decrementProductStoreLockWithCatch2(int goodsId, int buyNum);
}
5,OrderServiceImpl.java
@Service
public class OrderServiceImpl implements OrderService { @Resource
private GoodsMapper goodsMapper; /*
*
* 减库存,供其他方法调用
* */
private boolean decrestock(int goodsId, int buyNum) {
Goods goodsOne = goodsMapper.selectOneGoods(goodsId);
System.out.println("-------------------------当前库存:"+goodsOne.getStock()+"-------购买数量:"+buyNum);
if (goodsOne.getStock() < buyNum || goodsOne.getStock() <= 0) {
System.out.println("------------------------fail:buy fail,return");
return false;
}
int upStock = goodsOne.getStock()-buyNum;
goodsOne.setStock(upStock);
int upNum = goodsMapper.updateOneGoodsStock(goodsOne);
System.out.println("-------------------------success:成交订单数量:"+upNum); int insNum = goodsMapper.insertOneGoods(goodsOne);
System.out.println("-------------------------success:ins数量:"+insNum); return true;
} //方法内不做try catch捕捉异常,可以正常回滚
@Override
@Transactional(isolation = Isolation.REPEATABLE_READ)
public boolean decrementProductStoreLock(int goodsId, int buyNum) {
return decrestock(goodsId, buyNum);
} //异常时不处理,会导致不回滚
@Override
@Transactional(rollbackFor={Exception.class})
public boolean decrementProductStoreLockWithCatch(int goodsId, int buyNum) {
try {
return decrestock(goodsId, buyNum);
} catch (Exception e) {
System.out.println("捕捉到了异常in service method");
e.printStackTrace();
String errMsg = e.getMessage();
//throw e;
return false;
}
} //异常时手动抛出异常
@Override
@Transactional(rollbackFor={Exception.class})
public boolean decrementProductStoreLockWithCatch1(int goodsId, int buyNum) {
try {
return decrestock(goodsId, buyNum);
} catch (Exception e) {
System.out.println("捕捉到了异常in service method");
e.printStackTrace();
String errMsg = e.getMessage();
throw e;
}
} //异常时手动rollback
@Override
@Transactional(isolation = Isolation.REPEATABLE_READ)
public boolean decrementProductStoreLockWithCatch2(int goodsId, int buyNum) {
try {
return decrestock(goodsId, buyNum);
} catch (Exception e) {
System.out.println("捕捉到了异常in service method");
e.printStackTrace();
String errMsg = e.getMessage();
//手动rollback
TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
return false;
}
}
}
说明:decrestock这个方法中有两个写操作,分别是:减库存和insert一条商品记录,
后者会引发duplicate entry异常
6,GoodsMapper.xml
<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE mapper
PUBLIC "-//mybatis.org//DTD Mapper 3.0//EN"
"http://mybatis.org/dtd/mybatis-3-mapper.dtd">
<mapper namespace="com.transactional.demo.mapper.GoodsMapper">
<select id="selectOneGoods" parameterType="int" resultType="com.transactional.demo.pojo.Goods">
select * from goods where goodsId=#{goodsId}
</select> <update id="updateOneGoodsStock" parameterType="com.transactional.demo.pojo.Goods">
UPDATE goods SET
stock = #{stock}
WHERE goodsId = #{goodsId}
</update> <insert id="insertOneGoods" parameterType="com.transactional.demo.pojo.Goods" useGeneratedKeys="true" keyProperty="goodsId">
insert into goods(goodsName) values( #{goodsName})
</insert>
</mapper>
五,测试效果
1,测试正常回滚:访问:
http://127.0.0.1:8080/lock/lock
返回:
### Error updating database. Cause: java.sql.SQLIntegrityConstraintViolationException: Duplicate entry '100分电动牙刷' for key 'goodsName'
### The error may involve com.transactional.demo.mapper.GoodsMapper.insertOneGoods-Inline
### The error occurred while setting parameters
### SQL: insert into goods(goodsName) values( ?)
### Cause: java.sql.SQLIntegrityConstraintViolationException: Duplicate entry '100分电动牙刷' for key 'goodsName' ;
Duplicate entry '100分电动牙刷' for key 'goodsName';
nested exception is java.sql.SQLIntegrityConstraintViolationException: Duplicate entry '100分电动牙刷' for key 'goodsName'
查看数据表:
库存未减少,说明正常回滚
查看控制台:
-------------------------当前库存:100-------购买数量:1
-------------------------success:成交订单数量:1
捕捉到了异常 in controller
2,测试事务无效,不能正常回滚:访问:
http://127.0.0.1:8080/lock/lockcatch
返回:
false
查看数据表:
回滚失效
查看控制台:
-------------------------当前库存:100-------购买数量:1
-------------------------success:成交订单数量:1
捕捉到了异常in service method
3,测试捕捉到异常后手动抛出异常引发回滚:
访问:
http://127.0.0.1:8080/lock/lockcatch1
返回:
### Error updating database. Cause: java.sql.SQLIntegrityConstraintViolationException: Duplicate entry '100分电动牙刷' for key 'goodsName'
### The error may involve com.transactional.demo.mapper.GoodsMapper.insertOneGoods-Inline
### The error occurred while setting parameters
### SQL: insert into goods(goodsName) values( ?)
### Cause: java.sql.SQLIntegrityConstraintViolationException: Duplicate entry '100分电动牙刷' for key 'goodsName' ;
Duplicate entry '100分电动牙刷' for key 'goodsName';
nested exception is java.sql.SQLIntegrityConstraintViolationException: Duplicate entry '100分电动牙刷' for key 'goodsName'
查看数据表:
查看控制台:
-------------------------当前库存:99-------购买数量:1
-------------------------success:成交订单数量:1
捕捉到了异常in service method
...
捕捉到了异常 in controller
...
4,测试捕捉到异常后手动回滚
访问:
http://127.0.0.1:8080/lock/lockcatch2
返回:
false
查看数据表:
查看控制台:
-------------------------当前库存:99-------购买数量:1
-------------------------success:成交订单数量:1
捕捉到了异常in service method
5,大家注意最后两个返回的区别:为什么会不一样?
手动抛出异常后,会被controller捕捉到,所以没有返回success或false,
而是返回了异常的提示信息
六,查看spring boot的版本:
. ____ _ __ _ _
/\\ / ___'_ __ _ _(_)_ __ __ _ \ \ \ \
( ( )\___ | '_ | '_| | '_ \/ _` | \ \ \ \
\\/ ___)| |_)| | | | | || (_| | ) ) ) )
' |____| .__|_| |_|_| |_\__, | / / / /
=========|_|==============|___/=/_/_/_/
:: Spring Boot :: (v2.3.4.RELEASE)
spring boot:方法中使用try...catch导致@Transactional事务无效的解决(spring boot 2.3.4)的更多相关文章
- Node.js中针对中文的查找和替换无效的解决方法
Node.js中针对中文的查找和替换无效的解决方法. //tags的值: tag,测试,帖子 var pos1 = tags.indexOf("测"); //这里返回-1 ta ...
- 面试突击83:什么情况会导致@Transactional事务失效?
一个程序中不可能没有事务,而 Spring 中,事务的实现方式分为两种:编程式事务和声明式事务,又因为编程式事务实现相对麻烦,而声明式事务实现极其简单,所以在日常项目中,我们都会使用声明式事务 @Tr ...
- 【spring】在spring cloud项目中使用@ControllerAdvice做自定义异常拦截,无效 解决原因
之前在spring boot服务中使用@ControllerAdvice做自定义异常拦截,完全没有问题!!! GitHub源码地址: 但是现在在spring cloud中使用@ControllerAd ...
- Spring Boot+AngularJS中因为跨域导致Session丢失
http://blog.csdn.net/dalangzhonghangxing/article/details/52446821 如果还在为跨域问题烦恼,请查看博主的 解决angular+sprin ...
- Spring main方法中怎么调用Dao层和Service层的方法
在web环境中,一般serviceImpl中的dao之类的数据库连接都由容器启动的时候创建好了,不会报错.但是在main中,没有这个环境,所以需要获取环境: ApplicationContext ct ...
- iOS - UITableView中Cell重用机制导致Cell内容出错的解决办法
"UITableView" iOS开发中重量级的控件之一;在日常开发中我们大多数会选择自定Cell来满足自己开发中的需求, 但是有些时候Cell也是可以不自定义的(比如某一个简单的 ...
- 资料汇总--Ajax中Put和Delete请求传递参数无效的解决方法(Restful风格)【转】
开发环境:Tomcat9.0 在使用Ajax实现Restful的时候,有时候会出现无法Put.Delete请求参数无法传递到程序中的尴尬情况,此时我们可以有两种解决方案:1.使用地址重写的方法传递参数 ...
- Ajax中Put和Delete请求传递参数无效的解决方法(Restful风格)
本文装载自:http://blog.csdn.net/u012737182/article/details/52831008 感谢原文作者分享 开发环境:Tomcat9.0 在使用Ajax实现R ...
- jQuery中hover和blur使用代理delegate无效的解决方法
今天就遇到了这样的小问题: $(document).ready(function(){ $('.status_on').hover(function(){ $(this).html('点击禁用'); ...
随机推荐
- 吴恩达《深度学习》-第二门课 (Improving Deep Neural Networks:Hyperparameter tuning, Regularization and Optimization)-第一周:深度学习的实践层面 (Practical aspects of Deep Learning) -课程笔记
第一周:深度学习的实践层面 (Practical aspects of Deep Learning) 1.1 训练,验证,测试集(Train / Dev / Test sets) 创建新应用的过程中, ...
- 小BUG大原理:FastJSON实体转换首字母小写的尴尬事件
问题描述 因为项目连接的Oracle数据库,字段名映射方便使用大写,但是通过接口调用返回到前端的字段名首字母为小写,这样带来的问题前端显示的字段就需要写这种很尴尬的格式. 原因分析 开发环境使用的是S ...
- 口罩预约管理系统——数据库设计(前端+PHP+MySQL)
目录 一.背景 二.口罩预约管理系统介绍 三.数据库设计 四.MySQL创建数据库以及数据表 五.数据库设计总结 一.背景 2020年的疫情影响了我们的生产生活,政府不断加大力度联防联控,遏制疫情的蔓 ...
- MySQL安装错误Couldn't find MySQL server
Starting MySQL ERROR! Couldn't find MySQL server (/usr/local/mysql/bin/mysqld_safe) 昨天rpm安装MySQL5.7后 ...
- 面试官:哪些场景会产生OOM?怎么解决?
这个面试题是一个朋友在面试的时候碰到的,什么时候会抛出OutOfMemery异常呢?初看好像挺简单的,其实深究起来考察的是对整个JVM的了解,而且这个问题从网上可以翻到一些乱七八糟的答案,其实在总结下 ...
- PG-跨库操作-postgres_fdw
接上一篇<PG-跨库操作-dblink>:讲下postgres_fdw的使用:postgres_fdw工作原理详细介绍可以去看下<PostgreSQL指南>第4章: 对FDW特 ...
- OpenCV计算机视觉学习(1)——图像基本操作(图像视频读取,ROI区域截取,常用cv函数解释)
1,计算机眼中的图像 我们打开经典的 Lena图片,看看计算机是如何看待图片的: 我们点击图中的一个小格子,发现计算机会将其分为R,G,B三种通道.每个通道分别由一堆0~256之间的数字组成,那Ope ...
- pytest自学第一期
开始自学pytest了,我并不想看网上的各种自学教程和文档,要看咱们今天就看pytest的官方文档,不会英语咱们就用翻译,看不懂原理咱们就翻源码,就人肉试错 学习一个技术,使用速成鸡的套路是一个办法, ...
- Shell学习(三)Shell参数传递
一.传参实例 ##脚本文件内容 #执行的文件名 echo $0; #第一个参数 echo $1; #第二个参数 echo $2; #第三个参数 echo $3; ##调用语句 ./testShell. ...
- shiro 退出过滤器 logout ---退出清除HTTPSession数据
重写LogouFilter类 import org.apache.shiro.web.filter.authc.LogoutFilter; public class ShiroLogoutFilter ...