【springboot】事务处理
转自:
https://blog.csdn.net/cp026la/article/details/86496788
扯淡:
复杂的业务逻辑中一个请求可能需要多次操作数据库,要保证一个Service 方法中多个 dao 的操作同时成功(失败),事务的配置就很重要了。
大概分三种情况:
1、分布式事务:即多模块中事务,分布式事务建议是可以避免就避免,可以使用消息中间件处理,但也不能完全解决。
2、多线程事务:参考:https://blog.csdn.net/kongkongyanan/article/details/81703415
3、单模块中的事务,本章的重点,也是开发中遇到的最多的,这里给出两种配置方式。
本章分别使用注解 @Transactional (springboot默认推荐)和 AOP 全局配置的方式:
pom 依赖(延续上一章代码)增加aop的依赖:
<!-- aop -->
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
一、事务演示:
需要引入spring-boot-starter-jdbc依赖,但是我们集成了mybatis,已经包含了该依赖,不必重复引入。
1、controller层代码:在UserController中添加测试的方法
@GetMapping
@RequestMapping(value = "add/{number}")
public ApiResult add(@PathVariable("number") Integer number){
userService.addUser(number);
return ApiResult.ok();
}
2、service层代码:在UserService中添加测试方法,设置一个可控的除以0异常
public void addUser(int number){
User user = new User();
user.setId(22);
user.setName("xiaoming");
user.setAge(12);
mapper.insertSelective(user);
int num = 10/number;
User user2 = new User();
user2.setId(23);
user2.setName("xiaohong");
user2.setAge(90);
mapper.insertSelective(user2);
}
3、测试
3.1、请求接口:http://localhost:8080/user/add/10 数据库正常插入两条数据
3.2、删除刚插入的两条数据。请求接口:http://localhost:8080/user/add/0 制造除以0 异常。
现象:数据库此时只插入第一条数据(xiaoming)。
分析:插入第二条数据(xiaohong)之前,出现除以0异常,造成程序中断。
需求: 这种在同一个逻辑中的多个dao操作要么同时成功、要么同时失败,不允许只成功一个dao操作的情况出现。
二、使用@Transactional 配置事务:
1、在service层addUser 方法上添加@Transactional注解
@Transactional(rollbackFor = Exception.class)
public void addUser(int number){
...
}
2、测试:删除数据库中已经成功插入的数据。
2.1、请求接口:http://localhost:8080/user/add/0 制造除以0 异常。
2.2、结果:出现除以0 异常,数据库并未插入任何数据,即实现了多个dao操作同时成功,同时失败。
3、注意点:
1、spring 事务默认回滚的是运行时异常(RuntimeException)和错误(Error),非检测异常,例如SQLException 不会回滚。加上rollbackFor = Exception.class 可解决。
2、在service中使用了try catch 捕获了异常,事务不会回滚,因为try catch 异常之后就相当于没有异常,建议异常在controller中统一处理。
三、使用 AOP 全局处理事务:
需要引入 aop 的依赖。
1、新建全局事务配置类(个人放置到config包中,方便管理):
/**
* spring aop 配置全局事务
*/
@Aspect
@Configuration
public class TransactionAdviceConfig { /*
* 定义切入点
* execution()是最常用的切点函数
* execution (com.coolron.user.service.impl..*.*(..))
* 1、execution(): 表达式主体。
* 2、第一个*号:表示返回类型,*号表示所有的类型。
* 3、包名:表示需要拦截的包名,后面的两个句点表示当前包和当前包的所有子包,com.sample.service.impl包、子孙包下所有类的方法。
* 4、第二个*号:表示类名,*号表示所有的类。
* 5、*(..):最后这个星号表示方法名,*号表示所有的方法,后面括弧里面表示方法的参数,两个句点表示任何参数。
*/
private static final String AOP_POINTCUT_EXPRESSION = "execution(* com.coolron.user.service.impl..*.*(..))"; @Autowired
private PlatformTransactionManager transactionManager; @Bean
public TransactionInterceptor txAdvice() { DefaultTransactionAttribute txAttr_REQUIRED = new DefaultTransactionAttribute();
txAttr_REQUIRED.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED); DefaultTransactionAttribute txAttr_REQUIRED_READONLY = new DefaultTransactionAttribute();
txAttr_REQUIRED_READONLY.setPropagationBehavior(TransactionDefinition.PROPAGATION_REQUIRED);
txAttr_REQUIRED_READONLY.setReadOnly(true); NameMatchTransactionAttributeSource source = new NameMatchTransactionAttributeSource(); // service 中方法前缀
source.addTransactionalMethod("add*", txAttr_REQUIRED);
source.addTransactionalMethod("save*", txAttr_REQUIRED);
source.addTransactionalMethod("delete*", txAttr_REQUIRED);
source.addTransactionalMethod("update*", txAttr_REQUIRED); return new TransactionInterceptor(transactionManager, source);
} @Bean
public Advisor txAdviceAdvisor() {
AspectJExpressionPointcut pointcut = new AspectJExpressionPointcut();
pointcut.setExpression(AOP_POINTCUT_EXPRESSION);
return new DefaultPointcutAdvisor(pointcut, txAdvice());
} }
2、测试:
将 Service 中addUser 方法上的 @Transactional 注解注释,删除数据。
2.1、请求接口:http://localhost:8080/user/add/10 正常操作,数据库插入两条数据。
2.2、删除刚插入的两条数据。请求接口:http://localhost:8080/user/add/0 制造除以0 异常。
结果:出现异常,数据库并未插入数据。
3、注意点:
1、正确配置切入点的位置,本章配置的是:
execution(* com.coolron.user.service.impl….(…)),也可将user换成 * 指定多个位置
2、service 层中方法名应严格按照全局事务配置类中定义的规则来命名。
3、try catch 同样也不要在service中使用。
至此,单模块中的事务配置已完成,也是平时开发中最遇到的,对于分布式事务(即多模块),尽量将涉及到事务的业务逻辑放到一个模块中处理,使用消息中间件处理也不是最佳方案。
【springboot】事务处理的更多相关文章
- Springboot 事务处理常见坑点
使用事务注解@Transactional 之前,应该先了解它的相关属性,避免在实际项目中踩中各种各样的坑点. 常见坑点1:遇到非检测异常时,事务不开启,也无法回滚. 例如下面这段代码,账户余额依旧增加 ...
- SpringBoot学习笔记(9)----SpringBoot中使用关系型数据库以及事务处理
在实际的运用开发中,跟数据库之间的交互是必不可少的,SpringBoot也提供了两种跟数据库交互的方式. 1. 使用JdbcTemplate 在SpringBoot中提供了JdbcTemplate模板 ...
- springboot实现定时任务,异步操作,统一结果返回,全局异常处理,拦截器及事务处理
本文是作者原创,版权归作者所有.若要转载,请注明出处. 本文都是springboot的常用和实用功能,话不多说开始吧 定时任务 1.启动类开启注解 @EnableScheduling //开启基于注解 ...
- (十四)SpringBoot之事务处理
一.简介 ssh ssm都有事务管理service层通过applicationContext.xml配置,所有service方法都加上事务操作: 用来保证一致性,即service方法里的多个dao操作 ...
- Springboot全局事务处理
什么是全局事务 Spring Boot(Spring)事务是通过aop(aop相关术语:通知(Advice).连接点(Joinpoint).切入点(Pointcut).切面(Aspect).目标(Ta ...
- SpringBoot | 1.4 数据库事务处理
前言 前面讲解了Sring的AOP,可以知道它是用来抽取公共代码,增强方法的.而在JDBC操作数据库进行数据处理时,有很多重复的公共代码:事务的提交与回滚跟AOP的约定流程很相似.因此,Spring数 ...
- springboot多个service互相调用的事务处理(十三)
在一个service的方法A中,调用另一个service的方法B,方法A和方法B均存在数据库插入操作,需要添加如下配置: @Transactional(rollbackFor = Exception. ...
- Springboot数据访问,棒棒哒!
Springboot对数据访问部分提供了非常强大的集成,支持mysql,oracle等传统数据库的同时,也支持Redis,MongoDB等非关系型数据库,极大的简化了DAO的代码,尤其是Spring ...
- SpringBoot系列——Spring-Data-JPA
前言 jpa是ORM映射框架,更多详情,请戳:apring-data-jpa官网:http://spring.io/projects/spring-data-jpa,以及一篇优秀的博客:https:/ ...
随机推荐
- Django基础08篇 filter&tag
1.Django自带的过滤器filter views.py中代码 def template_tags(request): import datetime content = '三胖content三胖c ...
- Python运行时报错 ModuleNotFoundError: No module named ‘exceptions‘
踩的坑: 搜教程,很多文章都推荐使用:pip install python_docx‑0.8.10‑py2.py3‑none‑any.whl 但是依旧报错. 成功的示范: 使用命令:pip3 inst ...
- SFC style CSS variable injection
摘要 在单文件组件样式中支持使用组件状态驱动的 CSS 变量( CSS 自定义属性). 基础示例 <template> <div class="text"> ...
- Bat脚本备份sqlserver 表结构、存储过程、函数、指定表数据
Bat脚本备份sqlserver 表结构.存储过程.指定表数据: @echo off cd /d %~dp0 ::备份表结构.存储过程和部分配置表的数据 set LogFile=report.log ...
- Scala学习——函数
一.函数的定义(def) object FunctionApp { //定义函数:方法体内最后一行为返回值,不需要使用return def add(a:Int,b:Int):Int={ a + b } ...
- Qt 入门 ---- 如何在程序窗口显示图片?
步骤: 1. 选择资源(准备图片) 2. 加载资源(导入图片) 3. 使用资源(显示图片) 具体操作流程: ① 从网上寻找合适的图片素材,下载到本地,在项目根目录下创建一个images文件夹存储程序中 ...
- python 之爬虫基本流程
python 之爬虫基本流程 一 用户获取网络数据的方式: 方式1:浏览器提交请求--->下载网页代码--->解析成页面 方式2:模拟浏览器发送请求(获取网页代码)->提取有用的数据 ...
- tomcat与springmvc 结合 之---第17篇 StandContext容器和SpringMVC的WebApplicationContext的联系
writedby 张艳涛, 上一篇分析了,dispatcherservlet通过getServletConfig 方法获取了web.xml定义的<param-init>属性的过程 那么在如 ...
- 在 CentOS 7.5 64位上使用 yum 安装 MySQL 8.0
前段时间在 CentOS 7.5 64位上安装 MySQL 8.0.查了些资料,在这里记录一下详细的安装和设置步骤. 一.安装 使用yum安装MySQL之前需要先下载对应的.rpm文件,下载方法: 去 ...
- Intouch/ifix关于语音报警的一种设置思路
工控项目最近升级改造,需要使用Intouch/ifix提供一个语音报警功能.这个不像先前提供的单一的声音报警,业主方要求能详细的提供某某水泵或者是某某设备故障报警,这就要求我们这边对语音解析或者基础控 ...