2023.12.6 关于 Spring Boot 事务的基本概念

news2024/12/26 21:48:15

目录

事务基本概念

前置准备

Spring Boot 事务使用 

编程式事务

声明式事务

@Transactional 注解参数说明

@Transational 对异常的处理

解决方案一

解决方案二

@Transactional 的工作原理

面试题 

Spring Boot 事务失效的场景有那些?


事务基本概念

  • 事务指一组操作,这些操作要么全部成功,要么全部失败
  • 如果在一组操作中有一个操作失败了,那么整个事务便会回滚即撤销已经执行的操作,从而保证数据的一致性和完整性

实例理解

  • 典型实际场景为 银行转账操作
两个步骤
  1. 从源账户扣除指定金额
  2. 将该金额添加到目标账户
分析原因
  • 这两个步骤必须保证同时执行成功,如果其中任意一个步骤失败,便必须撤销整个操作,以保持数据的一致性
  • 即 在扣款成功后,如果存款时发生错误(如网络问题),那么我们必须要回滚扣款操作,以确保不会错误地从源账户中扣款

前置准备

  •  下述实例均基于 实现根据用户 id 删除用户信息功能

  • 创建一个 user 表,并添加几条用户信息

  • 创建 User 实体类 与 数据库的 user 表字段名相对应
import lombok.Data;

@Data
public class User {
    private int id;
    private String name;
    private int age;
    private String password;
    private int state;
}
  • 初始化 UserMapper 接口,此处我们添加一个 del 方法
import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Param;


//添加 @Mapper 注解 代表该接口会伴随这 项目的启动而注入到容器中
@Mapper
public interface UserMapper {

//    根据用户id 删除用户信息
    int del(@Param("user_id") int id);
}
  • 在与 接口相对应的 XML 文件中添加上与 del 方法 相对应的 sql 语句
<?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.example.demo.mapper.UserMapper">

    <delete id="del">
        delete from user where id = #{user_id}
    </delete>
</mapper>

Spring Boot 事务使用 

编程式事务

  • Spring Boot 中内置了两个对象
  • DataSourceTransactionManager 用来获取事务(开启事务)、提交或回滚事务
  • TransactionDefinition 为事务的属性,在获取事务的时候需要将其 传递进去,从而获得一个事务 TransactionStatus
实例理解
  • 我们在 UserController 中 使用编程式事务给 根据用户id 删除用户信息  这一功能加上事务
import com.example.demo.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.jdbc.datasource.DataSourceTransactionManager;
import org.springframework.stereotype.Controller;
import org.springframework.transaction.TransactionDefinition;
import org.springframework.transaction.TransactionStatus;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.ResponseBody;

@ResponseBody
@Controller
@RequestMapping("/user")
public class UserController {

    @Autowired
    private UserService userService;

//    编程式 事务
    @Autowired
    private DataSourceTransactionManager transactionManager;
    @Autowired
    private TransactionDefinition transactionDefinition;

    @RequestMapping("/del")
    public String del(Integer id) {
        if(id < 0 || id == null)  return "请输入正确的用户 id";
        TransactionStatus transactionStatus = null;
        int result = 0;
        try {
//            1. 开启事务
            transactionStatus = transactionManager.getTransaction(transactionDefinition);
//            2. 业务操作 删除用户
            result = userService.del(id);
            System.out.println("del 方法:" + (result == 1 ? "删除成功": "删除失败" ));
//            3. 提交事务
            transactionManager.commit(transactionStatus); // 提交事务
        }catch (Exception e) {
            if(transactionStatus != null){
//              发生异常 回滚事务
                transactionManager.rollback(transactionStatus); // 回滚事务
            }
        }
            return "del 方法:" + (result == 1 ? "删除成功": "删除失败" );
    }
}

测试结果

  • 在浏览器的 URL 地址框中输入相对应地址,来调用上述代码的方法


声明式事务

  • Spring Boot 提供了 @Transactional 注解实现事务
  • 只需在需要的方法上添加 @Transaction 注解即可
  • 无需手动开启事务和提交事务,进入方法时自动开启事务,方法执行完会自动提交事务
  • 如果中途发生了没有处理的异常会自动回滚事务

注意:

  • @Transactional 注解可以用来修饰方法或类
  • 修饰方式时,该方法必须为 public 否则不生效
  • 修饰类时,表明该注解对该类中所以的 public 方法都生效

实例理解

  • 我们在 UserController2 中 使用声明式事务给 根据用户id 删除用户信息  这一功能加上事务
import com.example.demo.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping("/user2")
public class UserController2 {
    @Autowired
    private UserService userService;
    
    @RequestMapping("/del")
    @Transactional
    public String del(Integer id) {
        if(id < 0 || id == null) return "请输入正确的用户 id";
        int result = userService.del(id);
        return "del 方法:" + (result == 1 ? "删除成功": "删除失败" );
    } 
}

测试结果

  • 在浏览器的 URL 地址框中输入相对应地址,来调用上述代码的方法


@Transactional 注解参数说明

参数作用
value当配置了多个事务管理器时,可以使用该属性指定选择哪个事务管理器
transactionManager当配置了多个事务管理器时,可以使用该属性指定选择哪个事务管理器
propagation事务的传播行为,默认值为 Propagation.REQUIRED
isolation事务的隔离级别,默认值为 Isolation.DEFAULT
timeout事务的超时时间,默认值为 -1 如果超过该事件限制但事务还没有完成则自动回滚事务
readOnly指定事务是否为只读事务 默认值为 false 为了忽略那些不需要事务的方法 比如读取事务
rollbackFor用于指定能够触发事务回滚的异常类型 可以指定多个异常类型
rollbackForClassName用于指定能够触发事务回滚的异常类型 可以指定多个异常类型
noRollbackFor抛出指定的异常类型,不会滚事务,也可以指定多个异常类型
noRollbackForClassName抛出指定的异常类型,不会滚事务,也可以指定多个异常类型

注意:

  • 区别 只读事务 和 无事务
  • 只读事务 可以设置隔离级别,默认为 repeatable read ,可设置 isolation 更改隔离级别

  • 无事务 仅为默认的隔离级别 repeatable read

@Transational 对异常的处理

实例理解
  • 此处我们故意在 UserController 中加入异常代码,并手动捕获该 算数异常
  • 那么此处 @Transational 是否会回滚 del 操作呢?
package com.example.demo.controller;

import com.example.demo.service.UserService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
@RequestMapping("/user2")
public class UserController2 {
    @Autowired
    private UserService userService;

    @RequestMapping("/del")
    @Transactional // 在方法执行前开启事务 方法正常执行完后提交事务 执行途中发生异常回滚事务
    public String del(Integer id) {
        if(id < 0 || id == null) return "请输入正确的用户 id";
        int result = userService.del(id);
        try {
            int num = 10/0;
        } catch (Exception e) {
            e.printStackTrace();
        }
        return "del 方法:" + (result == 1 ? "删除成功": "删除失败" );
    }
}

执行结果:

  • 在浏览器的 URL 地址框中输入相对应地址,来调用上述代码的方法

  • 由上图可知此处我们的 @Transational 并未回滚 del 操作
  • 即 @Transactional 在异常被捕获的情况下,不会进行事务的自动回滚
解决方案一
  • 捕获到异常后,再重新抛出异常,让框架感知到异常,如果框架感知到异常,便会自动回滚事务
@RequestMapping("/del")
@Transactional // 在方法执行前开启事务 方法正常执行完后提交事务 执行途中发生异常回滚事务public String del(Integer id) {
    if(id < 0 || id == null) return "请输入正确的用户 id";
    int result = userService.del(id);
    try {
        int num = 10/0;
    } catch (Exception e) {
        e.printStackTrace();
//        抛出异常
        throw e;
    }
    return "del 方法:" + (result == 1 ? "删除成功": "删除失败" );
}

执行结果:

  • 在浏览器的 URL 地址框中输入相对应地址,来调用上述代码的方法

  • 数据库中 id = 1 的 xiaolin 未被删除,说明此时 @Transational 进行了回滚操作
解决方案二
  • 捕获到异常后,手动回滚事务,此处框架是感知不到异常的
@RequestMapping("/del")
@Transactional // 在方法执行前开启事务 方法正常执行完后提交事务 执行途中发生异常回滚事务public String del(Integer id) {
    if(id < 0 || id == null) return "请输入正确的用户 id";
    int result = userService.del(id);
    try {
        int num = 10/0;
    } catch (Exception e) {
        e.printStackTrace();
//        手动回滚事务
        TransactionAspectSupport.currentTransactionStatus().setRollbackOnly();
    }
    return "del 方法:" + (result == 1 ? "删除成功": "删除失败" );
}
  • TransactionAspectSupport 为 Spring 框架中的一个类,提供了事务相关的支持和功能
  • currentTransactionStatus 为 TransactionAspectSupport 类的一个静态方法,用于获取当前事务的状态对象
  • setRollbackOnly 为 TransactionStatus 接口的一个方法,用于将当前事务标记为回滚状态0

执行结果:

  • 在浏览器的 URL 地址框中输入相对应地址,来调用上述代码的方法

  • 数据库中 id = 1 的 xiaolin 未被删除,说明此时 @Transational 进行了回滚操作

重点理解

  • 此处为什么会返回一个 删除成功?
  • 代码从上到下顺序执行,先执行了 del 操作
  • 即此处的 result 值已经成功被赋值为 1 (返回值为 del 操作影响的行数)
  • 然后我们才对 算数异常进行捕获,捕获之后再进行的 回滚操作
  • 且异常捕获之后,我们并未抛出异常,从而不会出现方案一的服务器错误
  • 在捕获完异常后代码将继续向下执行,此时便返回 "del 方法:" + (result == 1 ? "删除成功": "删除失败" )
  • 因为此处的 result 等于 1,所以返回了一个 删除成功
  • 但是我们要明白的是 我们在捕获异常后,在处理异常时进行了事务的回滚
  • 所以此处数据库中的  id = 1 的 xiaolin 未被删除

@Transactional 的工作原理

  • 此处声明式事务的实现方式 可使用 Spring AOP 来实现
  • 执行目标方法之前 先开启事务,类似于前置通知
  • 执行完目标方法之后 再提交事务,类似于后置通知
  • 如果在执行中途发生了没有处理的异常 便回滚事务
  • 综上 我们可以直接将目标方法 写入环绕通知中 
/*
* 环绕通知
* 此处的 joinPoint 就是连接点,即方法本身
* */
@Around("pointcut()")
public Object aroundAdvice(ProceedingJoinPoint joinPoint) throws Throwable {
    Object obj = null;
    System.out.println("执行目标方法之前 这里开启事务");
    try {
//        此处执行目标方法
        obj = joinPoint.proceed();
    }catch (Exception e) {
        System.out.println("执行目标方法出现异常 这里回滚事务");
    }
    System.out.println("执行目标方法之后 这里提交事务");
//    最后将执行的结果交给框架
    return obj;
}
  • 上述代码仅为 实现声明式事务 的大致思路

面试题 

Spring Boot 事务失效的场景有那些?

  1. @Transactional 修饰的方法为非 public ,导致事务失效
  2. @Transactional 设置了一个较小的超时时间,如果方法本身的执行时间超过了设置的 timeout 超时时间,那么就会导致本来应该正常插入数据的方法执行失败
  3. 代码中有 try/catch 语句,仅捕获异常,不进行额外处理,则导致 @Transactional  不会自动回滚事务
  4. 数据库不支持事务,我们程序中的 @Transactional 只是给调用的数据库发生了:开始事务、提交事务 或 回滚事务 的之类,但是如果数据库本身不支持事务,如 MySQL 中设置了使用 MySAM 引擎,那么它本身是不支持事务的,在这种情况下,即使在程序中添加了 @Transactional 注解,那么依然不会有事务行为
  5. 当调用类内部的 @Transactional 修饰的方法时,事务是不会生效的
@RequestMapping("/save")
public int saveMappping(UserInfo userInfo) {
    return save(userInfo);
}
@Transactional
public int save(UserInfo userInfo) {
    // 非空效验
    if (userInfo == null ||
        !StringUtils.hasLength(userInfo.getUsername()) ||
        !StringUtils.hasLength(userInfo.getPassword()))
        return 0;
    int result = userService.save(userInfo);
    int num = 10 / 0; // 此处设置一个异常
    return result;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1310593.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

10.CSS浮动

CSS浮动 1.介绍 在最初&#xff0c;浮动是用来实现文字环绕图片效果的&#xff0c;现在浮动是主流的页面布局方式之一 2.作用 让元素脱离标准流&#xff0c;同一级的浮动的元素可以并排在一排显示 3.元素浮动后的特点 脱离文档流不管浮动前是什么元素&#xff0c;浮动后&…

【为什么POI的SXSSFWorkbook占用内存更小?】

&#x1f513;为什么POI的SXSSFWorkbook占用内存更小&#xff1f; &#x1f3c6;POI的SXSSFWorkbook&#x1f3c6;POI的SXSSFWorkbook占用内存&#x1f3c6;扩展配置行缓存限制 &#x1f3c6;POI的SXSSFWorkbook SXSSFWorkbook类是Apache POI库的一部分&#xff0c;它是一个流…

产品入门第二讲:Axure产品元件库的使用

&#x1f4da;&#x1f4da; &#x1f3c5;我是默&#xff0c;一个在CSDN分享笔记的博主。&#x1f4da;&#x1f4da; ​​​​ &#x1f31f;在这里&#xff0c;我要推荐给大家我的专栏《Axure》。&#x1f3af;&#x1f3af; &#x1f680;无论你是编程小白&#xff0c;还是…

python下使用Open3D

1.切记不要安装最新的python否则无法使用open3D &#xff0c;官网显示只支持python3.8-3.11 这是我安装的python版本 2.由于访问github很慢&#xff0c;所以我手动下载ply文件 https://github.com/isl-org/open3d_downloads/releases/download/20220201-data/fragment.ply 3…

手写进度条,鼠标移入显示悬浮框

效果 <template><div class"box"><div class"mid-box"><div class"mid-contant"><!-- 提示框 --><divv-if"hover"class"tooltip":style"{top: hovertop,}"><div>{{ ho…

c语言堆排序(详解)

堆排序 堆排序是一种基于二叉堆数据结构的排序算法&#xff0c;它的基本概念包括&#xff1a; 建立堆&#xff1a;将待排序的列表构建成一个二叉堆&#xff0c;即满足堆的性质的完全二叉树&#xff0c;可以是最大堆或最小堆。最大堆要求父节点的值大于等于其子节点的值&#x…

Linux(21):软件安装 RPM,SRPM 与 YUM

软件管理员简介 以原始码的方式来安装软件&#xff0c;是利用厂商释出的Tarball来进行软件的安装。 不过&#xff0c;你每次安装软件都需要侦测操作系统与环境、设定编译参数、实际的编译、最后还要依据个人喜好的方式来安装软件到定位。这过程是真的很麻烦的。 如果厂商先在他…

FastAPI之表单数据

FastAPI 表单数据处理教程 FastAPI 是一个现代、快速&#xff08;高性能&#xff09;的 Web 框架&#xff0c;用于构建 API&#xff0c;它用 Python 3.6类型提示的特性旨在方便和快速地设计和构建 APIs&#xff0c;并且减少代码的冗余与错误。下面将介绍如何在 FastAPI 中处理…

工业级路由器在风力发电场的远程监控技术

工业级路由器在风力发电场的远程监控技术方面具有重要的应用意义。风力发电场通常由分布在广阔地区的风力发电机组组成&#xff0c;需要进行实时监测、数据采集和远程管理。工业级路由器作为网络通信设备&#xff0c;能够提供稳定可靠的网络连接和多种远程管理功能&#xff0c;…

鸿蒙OS应用开发之登录界面

在前面学习了输入文本组件和按钮组件,可以使用这两种组件来实现一些常用的功能,比如登录界面,这种界面是每个程序员经常会到遇到的,比如让用户输入用户名称和密码。 在这里我们就来实现如下面的界面: 在上面界面里,第一个文本框用来输入用户名称,第二个用来输入用户密码…

python列表的循环遍历

数据容器&#xff1a;一个可以存储多个元素的Python数据类型 有哪些数据容器&#xff1a;list&#xff08;列表&#xff09;&#xff0c;tuple&#xff08;元组&#xff09;&#xff0c;str&#xff08;字符串&#xff09;&#xff0c;set&#xff08;集合&#xff09;&#x…

【STM32】ADC模数转换器

1 ADC简介 ADC&#xff08;Analog-Digital Converter&#xff09;模拟-数字转换器 ADC可以将引脚上连续变化的模拟电压转换为内存中存储的数字变量&#xff0c;建立模拟电路到数字电路的桥梁 STM32是数字电路&#xff0c;只有高低电平&#xff0c;没有几V电压的概念&#xff…

21. python __init__.py 文件的行为

重复打印行为分析 说明结论主模块主模块所在位置不会被python认为是包 说明 我在调试代码的时候&#xff0c;发现上面的print打印了两次&#xff0c;如果将图片中的 from aaa.F import Cat 改成 from F import Cat 则print只会打印一次。这是为什么呢&#xff1f; 结论 from …

华为海思、燧原、海光等齐力打破封锁,谁主AI芯片江山| 百能云芯

近期&#xff0c;美国对英伟达出口进行了限制&#xff0c;导致英伟达无法向中国大陆销售AI芯片&#xff0c;这一局势催生了中国本土IC设计企业的崛起&#xff0c;包括华为旗下的海思科技、腾讯旗下的燧原科技&#xff0c;以及海光信息和新创公司天数智芯等纷纷抢占市场。 据百能…

微信公众服务号升级订阅号

服务号和订阅号有什么区别&#xff1f;服务号转为订阅号有哪些作用&#xff1f;首先我们要知道服务号和订阅号有什么区别。服务号侧重于对用户进行服务&#xff0c;每月可推送4次&#xff0c;每次最多8篇文章&#xff0c;发送的消息直接显示在好友列表中。订阅号更侧重于信息传…

深度探索Linux操作系统 —— Linux图形原理探讨

系列文章目录 深度探索Linux操作系统 —— 编译过程分析 深度探索Linux操作系统 —— 构建工具链 深度探索Linux操作系统 —— 构建内核 深度探索Linux操作系统 —— 构建initramfs 深度探索Linux操作系统 —— 从内核空间到用户空间 深度探索Linux操作系统 —— 构建根文件系统…

基于SpringBoot+Thymeleaf+Mybatis学生信息管理系统(源码+数据库)

一、项目简介 本项目是一套基于SpringBootThymeleafMybatis学生信息管理系统&#xff0c;主要针对计算机相关专业的正在做bishe的学生和需要项目实战练习的Java学习者。 包含&#xff1a;项目源码、数据库脚本等&#xff0c;该项目可以直接作为bishe使用。 项目都经过严格调试…

字符雨canvas

整体思路&#xff1a; 确定好字符雨的具体字符是什么&#xff0c;需要多少行多少列这里是写死的其实也可以用循环加随机的方式生成不一样的字符雨&#xff0c;行列也可以读一下宽度然后做一下出发算一下也行首先得有一张画布搞起&#xff0c;然后循环列数去绘画字符定时器循环…

SQL语句的执行顺序怎么理解?

SQL语句的执行顺序怎么理解&#xff1f; 我们常常会被SQL其书写顺序和执行顺序之间的差异所迷惑。理解这两者的区别&#xff0c;对于编写高效、可靠的SQL代码至关重要。今天&#xff0c;让我们用一些生动的例子和场景来深入探讨SQL的执行顺序。 一、书写顺序 VS 执行顺序 SQ…

数据结构和算法(全)

1.了解数据结构和算法 1.1 二分查找 二分查找&#xff08;Binary Search&#xff09;是一种在有序数组中查找特定元素的搜索算法。它的基本思想是将数组分成两半&#xff0c;然后比较目标值与中间元素的大小关系&#xff0c;从而确定应该在左半部分还是右半部分继续查找。这个…