java中实现事务的两种方式:编程式事务和声明式事务

涉及到与数据库交互就必须会用到事务,如果一个方法中需要用到事务的地方没有使用事务就会造成数据不一致的风险,进而导致比较严重的bug,比如扣款时,账户的余额已经进行了扣减但是相应的订单没有生成,这种涉及账目的问题如果不使用事务进行一致性控制后果会很严重。
在项目开发中,事务又可以分为单体事务和分布式事务,对于分布式系统要实现事务会比较复杂,有时候需要引入第三方系统控制一致性;而传统的单体应用就比较容易实现事务,尤其是使用到spring框架开发项目事务使用会更容易。
使用事务进行编程可以分为声明式事务和编程式事务,声明式事务只需要在需要事务控制的方法上面添加 @Transactional 注解就可以实现事务控制;编程式事务相对代码复杂一些,需要在代码中对异常进行捕获并控制事务的提交或回滚。下面就分别介绍一下如何在项目中使用这两种事务控制:
首先需要做一些准备工作:

  1. 引入相关依赖
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-jdbc</artifactId>
</dependency>
<dependency><groupId>org.mybatis.spring.boot</groupId><artifactId>mybatis-spring-boot-starter</artifactId><version>2.1.2</version>
</dependency>
<dependency><groupId>org.projectlombok</groupId><artifactId>lombok</artifactId><version>1.18.12</version><scope>provided</scope>
</dependency>
<dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId><version>5.1.48</version>
</dependency>
  1. 在application.yml中添加数据库相关的配置信息:
server:port: 8080shutdown: gracefulspring:application:name: test-dbdatasource:driver-class-name: com.mysql.jdbc.Driverurl: jdbc:mysql://localhost:3306/test?allowMultiQueries=true&useUnicode=true&characterEncoding=utf8&useSSL=false&serverTimezone=GMT%2B8username: rootpassword: 123456%Testhikari:auto-commit: trueconnection-timeout: 30000idle-timeout: 30000minimum-idle: 1maximum-pool-size: 4max-lifetime: 1800000connection-test-query: SELECT A FROM T_POOLPING
  1. 在数据库中创建表用于测试事务时使用:
-- ----------------------------
-- 学生测试表
-- ----------------------------
CREATE TABLE `student`  (`id` int NOT NULL AUTO_INCREMENT,`name` varchar(64) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NULL DEFAULT NULL,`score` smallint NULL DEFAULT NULL,PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 1 CHARACTER SET = utf8mb4 COLLATE = utf8mb4_0900_ai_ci ROW_FORMAT = Dynamic;-- ----------------------------
-- 用户测试表
-- ----------------------------
CREATE TABLE `user`  (`id` int NOT NULL AUTO_INCREMENT,`name` varchar(64) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NULL DEFAULT NULL,`age` smallint NULL DEFAULT NULL,`sex` tinyint NULL DEFAULT NULL,PRIMARY KEY (`id`) USING BTREE
) ENGINE = InnoDB AUTO_INCREMENT = 1 CHARACTER SET = utf8mb4 COLLATE = utf8mb4_0900_ai_ci ROW_FORMAT = Dynamic;-- ----------------------------
-- 连接测试表
-- ----------------------------
CREATE TABLE `T_POOLPING`  (`A` varchar(2) CHARACTER SET utf8mb4 COLLATE utf8mb4_0900_ai_ci NULL DEFAULT NULL
) ENGINE = InnoDB CHARACTER SET = utf8mb4 COLLATE = utf8mb4_0900_ai_ci ROW_FORMAT = Dynamic;
  1. 创建两个实体类,分别是学生信息和用户信息:
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;/*** @Author xingo* @Date 2024/1/30*/
@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class Student {private int id;private String name;private int score;
}
import lombok.AllArgsConstructor;
import lombok.Builder;
import lombok.Data;
import lombok.NoArgsConstructor;/*** @Author xingo* @Date 2024/1/30*/
@Data
@NoArgsConstructor
@AllArgsConstructor
@Builder
public class User {private int id;private String name;private int age;private int sex;
}
  1. 定义两个Mapper用于插入学生信息和用户信息:
import org.apache.ibatis.annotations.Insert;
import org.apache.ibatis.annotations.Mapper;/*** @Author xingo* @Date 2024/1/30*/
@Mapper
public interface StudentMapper {@Insert("insert into student(name, score) values(#{name}, #{score})")boolean insertStudent(Student student);
}
import org.apache.ibatis.annotations.Insert;
import org.apache.ibatis.annotations.Mapper;/*** @Author xingo* @Date 2024/1/30*/
@Mapper
public interface UserMapper {@Insert("insert into user(name, age, sex) values(#{name}, #{age}, #{sex})")boolean insertUser(User user);
}

以上准备工作完成后演示几种事务的使用:

一、声明式事务

声明式事务使用比较简单,在需要事务控制的方法上面添加 @Transactional 注解就可以实现事务控制,在spring中事务控制是通过aop实现的,所以调用方法必须要通过代理请求才能实现事务控制,并且方法内的异常一定要抛出不能在方法内捕获,否则就不能实现事务控制。

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.transaction.annotation.Transactional;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;/*** @Author xingo* @Date 2024/1/30*/
@RestController
public class DbController {@Autowiredprivate UserMapper userMapper;@Autowiredprivate StudentMapper studentMapper;@Transactional(rollbackFor = Exception.class)@GetMapping("/db/test1")public String testdb1(int num) {Student student = Student.builder().id(1).name("张三").score(100).build();User user = User.builder().id(1).name("张三").age(11).sex(1).build();userMapper.insertUser(user);System.out.println(1 / num);studentMapper.insertStudent(student);return "ok";}}

声明式事务使用非常方便,基本不需要额外的代码控制事务,但是它的控制粒度比较大,需要在方法上面进行控制,如果方法内部有比较耗时的操作将会导致事务不能提交,这会给数据库造成比较大的压力。要想让事务控制范围比较小的场景,就需要使用编程式事务实现:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.support.TransactionCallback;
import org.springframework.transaction.support.TransactionTemplate;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;import java.util.function.Consumer;/*** @Author xingo* @Date 2024/1/30*/
@RestController
public class DbController {@Autowiredprivate TransactionTemplate transactionTemplate;@Autowiredprivate UserMapper userMapper;@Autowiredprivate StudentMapper studentMapper;@GetMapping("/db/test2")public String testdb2(int num) {Student student = Student.builder().id(1).name("张三").score(100).build();User user = User.builder().id(1).name("张三").age(11).sex(1).build();// 有返回值的编程式事务transactionTemplate.execute(new TransactionCallback<Object>() {@Overridepublic Object doInTransaction(TransactionStatus status) {try {userMapper.insertUser(user);System.out.println(1 / num);studentMapper.insertStudent(student);return "ok";} catch (Exception e) {e.printStackTrace();status.setRollbackOnly();return null;}}});return "ok";}@GetMapping("/db/test3")public String testdb3(int num) {Student student = Student.builder().id(1).name("张三").score(100).build();User user = User.builder().id(1).name("张三").age(11).sex(1).build();// 没有返回值的编程式事务transactionTemplate.executeWithoutResult(new Consumer<TransactionStatus>() {@Overridepublic void accept(TransactionStatus status) {try {userMapper.insertUser(user);System.out.println(1 / num);studentMapper.insertStudent(student);} catch (Exception e) {e.printStackTrace();status.setRollbackOnly();}}});return "ok";}
}

上面使用 TransactionTemplate 是spring中建议使用的方式,也可以使用 TransactionManager 实现编程式事务,它对事务的实现更加灵活:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.transaction.PlatformTransactionManager;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.support.DefaultTransactionDefinition;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;/*** @Author xingo* @Date 2024/1/30*/
@RestController
public class DbController {@Autowiredprivate PlatformTransactionManager transactionManager;@Autowiredprivate UserMapper userMapper;@Autowiredprivate StudentMapper studentMapper;@GetMapping("/db/test4")public String testdb4(int num) {Student student = Student.builder().id(1).name("张三").score(100).build();User user = User.builder().id(1).name("张三").age(11).sex(1).build();// 定义事务,并设置隔离级别DefaultTransactionDefinition definition = new DefaultTransactionDefinition();definition.setPropagationBehavior(DefaultTransactionDefinition.PROPAGATION_REQUIRED);// 获取事务状态信息TransactionStatus status = transactionManager.getTransaction(definition);try {userMapper.insertUser(user);System.out.println(1 / num);studentMapper.insertStudent(student);// 提交事务transactionManager.commit(status);} catch (Exception e) {e.printStackTrace();// 回滚事务transactionManager.rollback(status);}return "ok";}
}

上面这些都是spring中提供的事务处理方式,它底层都是通过aop进行处理,再结合其他orm框架让代码比较容易开发和维护,但是归根到底与数据库交互还是要使用jdbc实现事务控制,下面展示如果使用jdbc原生方式实现事务控制的代码要怎么写:

import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;import javax.sql.DataSource;
import java.sql.Connection;
import java.sql.PreparedStatement;/*** @Author xingo* @Date 2024/1/30*/
@RestController
public class DbController {@Autowiredprivate DataSource dataSource;@Autowiredprivate UserMapper userMapper;@Autowiredprivate StudentMapper studentMapper;@GetMapping("/db/test5")public String testdb5(int num) throws Exception {Student student = Student.builder().id(1).name("张三").score(100).build();User user = User.builder().id(1).name("张三").age(11).sex(1).build();Connection conn = dataSource.getConnection();try {conn.setAutoCommit(false);PreparedStatement pstmt1 = conn.prepareStatement("insert into user(name, age, sex) values(?, ?, ?)");pstmt1.setString(1, user.getName());pstmt1.setInt(2, user.getAge());pstmt1.setInt(3, user.getSex());pstmt1.executeUpdate();System.out.println(1 / num);PreparedStatement pstmt2 = conn.prepareStatement("insert into student(name, score) values(?, ?)");pstmt2.setString(1, student.getName());pstmt2.setInt(2, student.getScore());pstmt2.executeUpdate();conn.commit();} catch (Exception e) {e.printStackTrace();conn.rollback();} finally {conn.setAutoCommit(true);conn.close();}return "ok";}
}

上面示例的所有接口可以模拟正常请求和异常请求两种方式,通过模拟产生除数为0的异常来验证事务失效的场景:

# 正常请求
http://localhost:8080/db/test1?num=1# 异常请求
http://localhost:8080/db/test1?num=0

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

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

相关文章

方法阻塞的解决方案之一

1、简单使用 一个h一个cpp文件 #pragma once #include <iostream> #include <thread> #include <atomic> #include <chrono> #include <string>class Person {public:struct dog {std::string name;int age;};public:void a(std::atomic<bo…

设计模式篇---备忘录模式

文章目录 概念结构实例总结 概念 备忘录模式&#xff1a;在不破坏封装的前提下捕获一个对象的内部状态&#xff0c;并在该对象之外保存这个状态&#xff0c;像这样可以在以后将对象恢复到原先保存的状态。 就好比我们下象棋&#xff0c;下完之后发现走错了&#xff0c;想要回退…

【C++】美感(beautiful)

题目描述 寿寿喜欢有美感的序列。 对于寿寿来说一个序列是有美感的&#xff0c;当且仅当这个序列每两个相邻的数的和是m的倍数。特别的&#xff0c;寿寿认为长度为1的序列也是具有美感的。 寿寿现在随便在纸上写了一个长度为n的序列a&#xff0c;这个序列目前还不具有美感。因此…

Unity 自动轮播、滑动轮播

如图所示&#xff0c;可设置轮播间隔&#xff0c;可左右滑动进行轮播 1.在UGUI创建个Image&#xff0c;添加自动水平组件 2.添加并配置脚本 3.代码如下&#xff0c;都有注释 using UnityEngine; using UnityEngine.UI;public class IndicatorManager : MonoBehaviour {public …

【乳腺肿瘤诊断分类及预测】基于LVQNN学习向量量化神经网络

课题名称&#xff1a;基于LVQ神经网络的乳腺肿瘤诊断&#xff08;类型分类&#xff09; 版本日期&#xff1a;2023-03-10 运行方式: 直接运行0501_LVQ0501.m 文件即可 代码获取方式&#xff1a;私信博主或QQ&#xff1a;491052175 模型描述&#xff1a; 威斯康辛大学医学院…

(HAL)STM32F407ZGT6——10-4 高级定时器 PWM 输入模式实验

一、高级定时器简介 高级定时器的框图和通用定时器框图很类似&#xff0c;只是添加了其它的一些功能&#xff0c;如&#xff1a;重复计数器、带死区控制的互补输出通道、断路输入等。 高级定时器的时钟来自APB2, 而PCLK2 168Mhz, 我们设置PPRE2不分频, 因此高级定时器时钟 …

《微信小程序开发从入门到实战》学习九十九

7.4 视频组件 7.4.1 video组件 支持属性属性如下&#xff1a; 属性类型默认值说明最低版本src&#xff08;必填&#xff09;string要播放视频的资源地址&#xff0c;基础库2.3.0版本开始支持云文件ID1.0.0controlsbooleantrue是否显示默认播放控件&#xff08;播放/暂停按钮…

vue/js 调用系统打印功能进行图片(imgUrl,base64),表格,自定义内容,页面局部区域打印【print-js、html2canvas】

1.打印图片(imgUrl) <template><div><button click"jsonPrint">打印</button></div> </template><script lang"ts"> import printJS from "print-js"; export default {setup() {function jsonPrint(…

使用.NET6 Avalonia开发跨平台三维应用

本文介绍在Vistual Studio 2022中使用Avalonia和集成AnyCAD Rapid AvaloniaUI三维控件的过程。 0 初始化环境 安装Avalonia.Templates dotnet new install Avalonia.Templates若之前安装过可忽略此步骤。 1 创建项目 选择创建AvaloniaUI项目 选一下.NET6版本和Avalonia版…

detectron2的read_image方法

在看代码的时候&#xff0c;看到一行注释&#xff1a;use PIL, to be consistent with evaluation 说是用PIL方法加载&#xff0c;却又看见了BGR这种表述&#xff0c;后面的调用也都是cv2格式&#xff1a; 那我就要看下这里面是怎么实现的了&#xff0c;找到了read_image函数&…

[R] Why data manipulation is crucial and sensitive?

What does a data scientist really do? Identifying the pattern in cultural consumption, making fancy graph, engage a dialogue between data and the existing literature, refining hypothesis….(done within one months with three to four online meetings with p…

app的启动

前言 本篇文章讲解ios的应用程序的启动 应用程序的加载 点击一个app 首先&#xff0c;我们在手机上点击一个app图标 内核初始化 操作系统收到启动app的消息后&#xff0c;会调用内核代码初始化内存空间&#xff0c;为app创建进程然后操作系统通过系统调用读取并解析app的…

apktool 简单快速 反编译apk获取图片资源

apktool:下载地址&#xff1a;iBotPeaches / Apktool / Downloads — Bitbucket把 myapp.apk 和 apktool_2.9.3.jar 放在同一文件夹&#xff0c;注意不要有中文路径 java -jar apktool_2.9.3.jar d -f myapp.apk -o myapp java -jar: java 执行jar命令 apktool_2.9.3.jar: a…

C语言实现的数组合并与排序程序

引言 在本篇博客中&#xff0c;我们将详细解析一段C语言代码&#xff0c;该代码实现了从用户处接收两个整数数组&#xff0c;将它们合并为一个数组后进行排序&#xff0c;并最终输出排序后的结果。这段代码主要涵盖了数组操作、数据输入、冒泡排序算法以及数据输出等核心编程概…

【涵子来信】——拆机,感想

大家好&#xff0c;我是涵子。 初中的第一个学期结束了&#xff0c;来临寒假。我在寒假做了一件有趣的事情&#xff1a;拆机&#xff0c;修手机。今天我来分享分享这件事情。 拆机 情况介绍 拆机对象&#xff1a; iPhone 6 Plus 情况&#xff1a; 电池健康度100%&#xff08…

按身高和体重排队(100%用例)C卷(JavaPythonC++Node.jsC语言)

某学校举行运动会,学生们按编号 (1 、 2 、 3 … n) 进行标识,现需要按照身高由低到高排列,对身高相同的人,按体重由轻到重排列;对于身高体重都相同的人,维持原有的编号顺序关系。请输出排列后的学生编号。 输入描述: 两个序列,每个序列由n个正整数组成(0 < n <…

Unity-WebGL

问题&#xff1a;提示gzip压缩报错解决&#xff1a;关闭打包的地方压缩&#xff0c;如下图问题&#xff1a;窗口未全屏解决&#xff1a;使用百分比画布替换固定尺寸画布 参考&#xff1a;新版Unity打包Webgl端进行屏幕自适应_unity webgl分辨率自适应-CSDN博客问题&#xff1a;…

GBASE数据库注册例程的权限

要在数据库中注册例程&#xff0c;被授权的用户将 SPL 命令包含在 CREATE FUNCTION 或 CREATE PROCEDURE 语句中。数据库服务器存储内部注册了的 SPL 例程。下列用户具 有在数据库中注册新的例程的资格&#xff1a; • 有 DBA 权限的任何用户可在 CREATE 语句中&#xff0c;使…

Springboot+Redis

首先前提我们要在自己的本机电脑或者服务器上安装一个redis的服务器 Redis配置 添加依赖: <!-- SpringBoot Boot Redis --> <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-redis</artif…

MicroPython核心:编译器

MicroPython编译过程包括以下步骤&#xff1a; 词法分析器将MicroPython程序文本流转换为标记。语法解释器将标记转换为抽象语法&#xff08;语法树&#xff09;。根据语法书输出字节码或本地代码。 本文以给MicroPython增加一个简单的语言特性为例来说明这一过程&#xff1a…