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

相关文章

SpringBoot结合Docker进行容器化处理指南

《SpringBoot结合Docker进行容器化处理指南》在当今快速发展的软件工程领域,SpringBoot和Docker已经成为现代Java开发者的必备工具,本文将深入讲解如何将一个SpringBo... 目录前言一、为什么选择 Spring Bootjavascript + docker1. 快速部署与

Spring Boot spring-boot-maven-plugin 参数配置详解(最新推荐)

《SpringBootspring-boot-maven-plugin参数配置详解(最新推荐)》文章介绍了SpringBootMaven插件的5个核心目标(repackage、run、start... 目录一 spring-boot-maven-plugin 插件的5个Goals二 应用场景1 重新打包应用

SpringBoot+EasyExcel实现自定义复杂样式导入导出

《SpringBoot+EasyExcel实现自定义复杂样式导入导出》这篇文章主要为大家详细介绍了SpringBoot如何结果EasyExcel实现自定义复杂样式导入导出功能,文中的示例代码讲解详细,... 目录安装处理自定义导出复杂场景1、列不固定,动态列2、动态下拉3、自定义锁定行/列,添加密码4、合并

Spring Boot集成Druid实现数据源管理与监控的详细步骤

《SpringBoot集成Druid实现数据源管理与监控的详细步骤》本文介绍如何在SpringBoot项目中集成Druid数据库连接池,包括环境搭建、Maven依赖配置、SpringBoot配置文件... 目录1. 引言1.1 环境准备1.2 Druid介绍2. 配置Druid连接池3. 查看Druid监控

Java中读取YAML文件配置信息常见问题及解决方法

《Java中读取YAML文件配置信息常见问题及解决方法》:本文主要介绍Java中读取YAML文件配置信息常见问题及解决方法,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要... 目录1 使用Spring Boot的@ConfigurationProperties2. 使用@Valu

创建Java keystore文件的完整指南及详细步骤

《创建Javakeystore文件的完整指南及详细步骤》本文详解Java中keystore的创建与配置,涵盖私钥管理、自签名与CA证书生成、SSL/TLS应用,强调安全存储及验证机制,确保通信加密和... 目录1. 秘密键(私钥)的理解与管理私钥的定义与重要性私钥的管理策略私钥的生成与存储2. 证书的创建与

浅析Spring如何控制Bean的加载顺序

《浅析Spring如何控制Bean的加载顺序》在大多数情况下,我们不需要手动控制Bean的加载顺序,因为Spring的IoC容器足够智能,但在某些特殊场景下,这种隐式的依赖关系可能不存在,下面我们就来... 目录核心原则:依赖驱动加载手动控制 Bean 加载顺序的方法方法 1:使用@DependsOn(最直

SpringBoot中如何使用Assert进行断言校验

《SpringBoot中如何使用Assert进行断言校验》Java提供了内置的assert机制,而Spring框架也提供了更强大的Assert工具类来帮助开发者进行参数校验和状态检查,下... 目录前言一、Java 原生assert简介1.1 使用方式1.2 示例代码1.3 优缺点分析二、Spring Fr

java使用protobuf-maven-plugin的插件编译proto文件详解

《java使用protobuf-maven-plugin的插件编译proto文件详解》:本文主要介绍java使用protobuf-maven-plugin的插件编译proto文件,具有很好的参考价... 目录protobuf文件作为数据传输和存储的协议主要介绍在Java使用maven编译proto文件的插件

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class