Spring 事务及事务传播机制(1)

2024-05-09 08:52
文章标签 java spring 机制 事务 传播

本文主要是介绍Spring 事务及事务传播机制(1),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

目录

事务

回顾: 什么是事务

为什么需要事务

事务的操作

Spring事务的实现

Spring编程式事务(简单了解即可, 问就是基本不用)

 观察事务提交

观察事务回滚

Spring声明式事务 @Transactional

@Transactional作用


事务

回顾: 什么是事务

定义: 事务是指逻辑上的一组操作, 构成这组操作的各个单元, 要么全部执行, 要么全部不执行.

为什么需要事务

一般在进行程序设计中, 我们一般都会用到事务.

比如当家人给我们转账(2000元)时, 共有两个操作: 家人扣款2000, 我们的账户增加2000元.

但在家人扣款之后, 出现了一个问题: 此时一位自信的挖掘机操作员把光缆给"不小心"挖断了.

然后这就导致我们账户增加余额的操作得不到落实 .

因此这里引入事务: 使得家人扣款和我们账户到账是同一个事务, 这时如果中间执行这个事务时,就会发生异常, 然后就会回滚, 恢复数据: 家人账户+2000.

事务的操作

事务的操作主要有三步:

1.开启事务start transaction(在一组操作之前开启事务).

2.提交事务: commit.(这组操作全部执行成功, 就会全部落实)

3.回滚: rollback(这中间任意一个操作出现异常, 就会回滚事务, 即恢复初始状态)  

Spring事务的实现

Spring中的事务操作分为两类:

1.编程式事务(手动写代码操作事务)

2.声明式事务(利用注解操作事务)

我们通过一个简单的小项目来说明这个事务:

Spring编程式事务(简单了解即可, 问就是基本不用)

Spring手动操作事务和之前mysql操作事务类似.,有三个重要的操作步骤:

开启事务(获取事务)

提交事务

回滚事务 

SpringBoot内置了两个对象:

1. DataSourceTransactionManager 事务管理器. 用来获取事务(开启事务), 提交或回滚事务.

2.TransactionDefinition是事务的属性, 在获取事务时需要将TransactionDefinition传递进去获取一个TransactionStatus.

还是通过代码练习(就以一个简单的注册提交为示例, 很简单, 自行编写即可(作者懒得粘)):

@RestController
@RequestMapping("/user")
public class UserController {//JDBC事务管理器@Autowiredprivate DataSourceTransactionManager dataSourceTransactionManager;//定义事务属性@Autowiredprivate TransactionDefinition transactionDefinition;@Autowiredprivate UserService userService;@RequestMapping("/registry")public String registry(String name, String password) {//开启事务TransactionStatus transactionStatus = dataSourceTransactionManager.getTransaction(transactionDefinition);//用户这侧userService.registryUser(name, password);//提交事务dataSourceTransactionManager.commit(transactionStatus);//回滚事务//dataSourceTransactionManager.rollback(transactionStatus);return "注册成功";}
}

 观察事务提交

运行程序: http://127.0.0.1:8080/user/registry?name=admin&password=admin

观察数据库的结果, 数据插入成功.

观察事务回滚

运行程序:

 

虽然这里显示"注册成功", 但是数据库中并没有新增数据.

 

以上通过编程方法确实能够实现事务, 但是有没有什么更简单的方法呢?

Spring声明式事务 @Transactional

声明式事务很简单, 只需要在事务的方法上添加@Transactional注解就可以实现了. 无需手动开启事务和提交事务, 进入方法时自动开启事务, 方法执行完会自动提交事务, 如果中途发生了没有处理的异常会自动回滚事务.

我们来看代码实现:

@RestController
@RequestMapping("/user")
public class UserController1 {@Autowiredprivate UserService1 userService;@Transactional@RequestMapping("/registry")public String registry(String name, String password) {//用户注册userService.registryUser(name, password);return "注册成功";}
}

运行程序, 发现数据插入成功(通过这个id情况我们也可以知道之前那个事务中的内容确实是执行了, 但是回滚了):

 

修改程序, 使之出现异常:

@Slf4j
@RestController
@RequestMapping("/user")
public class UserController1 {@Autowiredprivate UserService1 userService;@Transactional@RequestMapping("/registry")public String registry(String name, String password) {//用户注册userService.registryUser(name, password);log.info("用户数据插入成功");//强制程序发生异常int a = 10 / 0;return "注册成功";}
}

观察:

 

观察后端: 虽然日志显示数据插入成功, 但是数据库却没有新增数据, 这证明了事务进行了回滚. 

@Transactional作用

@Transactional 可以用来修饰方法或类:

修饰方法(推荐)时: 只有修饰public方法时才生效(修饰其它方法时不报错, 也不生效).

修饰类时: 对@Transactional修饰类中所有的public方法都生效.

方法/类被@Transactional修饰时, 在目标方法执行开始之前, 会自动开启事务, 方法执行结束之后, 自动提交事务(即程序运行成功时, 自动提交).

那么在程序出现异常时什么时候会进行事务回滚, 什么时候不进行事务回滚呢? 这就比较复杂了.

让我们通过一张图看一下:

我们对之前的代码进行异常捕获再试一下:

我们发现: 虽然程序出错了, 但是由于异常被捕获了, 所以事务依然得到了提交. 

1.重新抛出异常以进行事务回滚:

//对异常进行捕获
try {//强制程序发生异常int a = 10 / 0;
} catch (Exception e) {//再次抛出异常throw e;
}

 2.手动回滚事务

//对异常进行捕获
try {//强制程序发生异常int a = 10 / 0;
} catch (Exception e) {//手动进行回滚TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
}

这篇关于Spring 事务及事务传播机制(1)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/972949

相关文章

Java中Map的五种遍历方式实现与对比

《Java中Map的五种遍历方式实现与对比》其实Map遍历藏着多种玩法,有的优雅简洁,有的性能拉满,今天咱们盘一盘这些进阶偏基础的遍历方式,告别重复又臃肿的代码,感兴趣的小伙伴可以了解下... 目录一、先搞懂:Map遍历的核心目标二、几种遍历方式的对比1. 传统EntrySet遍历(最通用)2. Lambd

Spring Boot 中 RestTemplate 的核心用法指南

《SpringBoot中RestTemplate的核心用法指南》本文详细介绍了RestTemplate的使用,包括基础用法、进阶配置技巧、实战案例以及最佳实践建议,通过一个腾讯地图路线规划的案... 目录一、环境准备二、基础用法全解析1. GET 请求的三种姿势2. POST 请求深度实践三、进阶配置技巧1

springboot+redis实现订单过期(超时取消)功能的方法详解

《springboot+redis实现订单过期(超时取消)功能的方法详解》在SpringBoot中使用Redis实现订单过期(超时取消)功能,有多种成熟方案,本文为大家整理了几个详细方法,文中的示例代... 目录一、Redis键过期回调方案(推荐)1. 配置Redis监听器2. 监听键过期事件3. Redi

Spring Boot 处理带文件表单的方式汇总

《SpringBoot处理带文件表单的方式汇总》本文详细介绍了六种处理文件上传的方式,包括@RequestParam、@RequestPart、@ModelAttribute、@ModelAttr... 目录方式 1:@RequestParam接收文件后端代码前端代码特点方式 2:@RequestPart接

SpringBoot整合Zuul全过程

《SpringBoot整合Zuul全过程》Zuul网关是微服务架构中的重要组件,具备统一入口、鉴权校验、动态路由等功能,它通过配置文件进行灵活的路由和过滤器设置,支持Hystrix进行容错处理,还提供... 目录Zuul网关的作用Zuul网关的应用1、网关访问方式2、网关依赖注入3、网关启动器4、网关全局变

SpringBoot全局异常拦截与自定义错误页面实现过程解读

《SpringBoot全局异常拦截与自定义错误页面实现过程解读》本文介绍了SpringBoot中全局异常拦截与自定义错误页面的实现方法,包括异常的分类、SpringBoot默认异常处理机制、全局异常拦... 目录一、引言二、Spring Boot异常处理基础2.1 异常的分类2.2 Spring Boot默

基于SpringBoot实现分布式锁的三种方法

《基于SpringBoot实现分布式锁的三种方法》这篇文章主要为大家详细介绍了基于SpringBoot实现分布式锁的三种方法,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、基于Redis原生命令实现分布式锁1. 基础版Redis分布式锁2. 可重入锁实现二、使用Redisso

SpringBoot的全局异常拦截实践过程

《SpringBoot的全局异常拦截实践过程》SpringBoot中使用@ControllerAdvice和@ExceptionHandler实现全局异常拦截,@RestControllerAdvic... 目录@RestControllerAdvice@ResponseStatus(...)@Except

Springboot配置文件相关语法及读取方式详解

《Springboot配置文件相关语法及读取方式详解》本文主要介绍了SpringBoot中的两种配置文件形式,即.properties文件和.yml/.yaml文件,详细讲解了这两种文件的语法和读取方... 目录配置文件的形式语法1、key-value形式2、数组形式读取方式1、通过@value注解2、通过

Java 接口定义变量的示例代码

《Java接口定义变量的示例代码》文章介绍了Java接口中的变量和方法,接口中的变量必须是publicstaticfinal的,用于定义常量,而方法默认是publicabstract的,必须由实现类... 在 Java 中,接口是一种抽象类型,用于定义类必须实现的方法。接口可以包含常量和方法,但不能包含实例