Spring Boot+Atomikos进行多数据源的分布式事务管理详解和实例

文章目录

  • 0.前言
  • 1.参考文档
  • 2.基础介绍
  • 3.步骤
      • 1. 添加依赖到你的`pom.xml`文件:
      • 2. 配置数据源及其对应的JPA实体管理器和事务管理器:
      • 3. Spring Boot+MyBatis集成Atomikos
      • 4. 在application.properties文件中配置数据源和JPA属性:
  • 4.使用示例
  • 5.底层原理

在这里插入图片描述

0.前言

背景: 一直零散的使用着Spring Boot 的各种组件和特性,从未系统性的学习和总结,本次借着这个机会搞一波。共同学习,一起进步。哈哈

Atomikos是一个易用、可靠、开放源码的事务管理器,它可以用于管理分布式事务,尤其在微服务架构中非常实用。它支持JTA(Java Transaction API)规范,能够与各种JTA兼容的资源管理器(如数据库和消息队列)配合使用。

  1. 分布式事务:当一个业务操作需要修改多个资源(如多个数据库或消息队列)时,我们需要保证这些修改操作的原子性,即它们要么全部成功,要么全部失败。这就是分布式事务。

  2. JTA:Java Transaction API(JTA)是Java平台的一个事务规范,定义了用户和事务管理器以及事务管理器和资源管理器之间的接口。Atomikos作为一个事务管理器,就是遵循JTA规范的。

  3. XA协议:XA协议是分布式事务的一个重要协议,它定义了全局事务ID、分支事务ID等概念,以及如何协调分支事务的接口。Atomikos支持XA协议。

Atomikos还提供了自动恢复、故障转移等高级特性,以进一步提高分布式事务的可靠性。

1.参考文档

Spring Boot 提供了一个用于整合 Atomikos 的 starter,名为 spring-boot-starter-jta-atomikos。它是 Spring Boot 提供的一系列 “starter” 依赖之一

  1. Spring Boot 官方文档
    Spring Boot 官方文档的 “Spring Boot Features” 部分有一个 “Working with JTA” 的小节,其中提到了如何使用 spring-boot-starter-jta-atomikos。链接:https://docs.spring.io/spring-boot/docs/2.5.3/reference/htmlsingle/#boot-features-jta
    在这里插入图片描述

  2. Atomikos 官方文档
    虽然 Atomikos 的官方文档并没有专门介绍 spring-boot-starter-jta-atomikos,但它提供了一些关于如何使用 Atomikos 的教程,你可以参考这些教程来理解 spring-boot-starter-jta-atomikos 是如何工作的。链接:https://www.atomikos.com/Documentation/SpringBootIntegration

2.基础介绍

Atomikos 是一个提供分布式事务管理的开源事务管理器。将它们结合使用,可以在 Spring Boot 应用程序中实现分布式事务的管理。

在 Spring Boot 中使用 Atomikos,通常需要进行以下步骤:

  1. 引入 Atomikos 依赖:
    首先,在 Maven 或 Gradle 构建文件中添加 Atomikos 的依赖项。可以添加 atomikos-transactions-spring-boot-starter 依赖,它是 Atomikos 与 Spring Boot 集成的起点。

  2. 配置数据源:
    在 Spring Boot 应用程序中,你需要配置多个数据源。可以使用 Spring Boot 的自动配置功能,根据配置文件或属性来配置数据源。你可以使用任何支持 Atomikos 的数据源,如 Atomikos 提供的 AtomikosDataSourceBean 或其他第三方数据源。

  3. 配置 Atomikos 事务管理器:
    在 Spring Boot 应用程序中,你需要配置 Atomikos 事务管理器。可以通过在配置类中创建 JtaTransactionManager 实例,并将其与 Atomikos 的 UserTransactionManagerTransactionManager 关联起来。这样,Spring 将使用 Atomikos 事务管理器来管理分布式事务。

  4. 配置 JTA 事务管理器:
    为了使 Spring Boot 应用程序能够使用 Atomikos 进行分布式事务管理,你需要配置 JTA 事务管理器。可以使用 Spring Boot 的自动配置功能,根据配置文件或属性来配置 JTA 事务管理器。

  5. 在方法上添加 @Transactional 注解:
    在需要进行事务管理的方法上,添加 Spring 的 @Transactional 注解。这将告诉 Spring 在方法执行期间启动和提交事务,并回滚事务(如果发生异常)。

使用 Spring Boot+Atomikos 的原理如下:

  1. Spring Boot 提供了自动配置功能,可以根据配置文件或属性来自动配置数据源和事务管理器。

  2. Atomikos 是一个独立的事务管理器,它提供了 JTA(Java Transaction API)的实现,可以处理分布式事务。

  3. 在 Spring Boot 中,你配置了多个数据源,并使用 Atomikos 的数据源实现(如 AtomikosDataSourceBean)来实现分布式事务。

  4. 当你在方法上添加了 @Transactional 注解时,Spring Boot 会使用 Atomikos 的事务管理器来管理事务。

  5. 当方法执行时,事务管理器会协调各个数据源的事务,并在方法执行完成后根据事务的状态来提交或回滚事务。

总结来说,Spring Boot+Atomikos 的原理是利用 Spring Boot 的自动配置功能来配置多个数据源和事务管理器,并使用 Atomikos 的事务管理器实现分布式事务的管理。这样,你可以在 Spring Boot 应用程序中使用 @Transactional 注解来管理分布式事务。

3.步骤

确保你的数据库支持XA事务,否则无法使用Atomikos。

1. 添加依赖到你的pom.xml文件:

<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-jta-atomikos</artifactId>
</dependency>
<dependency><groupId>mysql</groupId><artifactId>mysql-connector-java</artifactId>
</dependency>

2. 配置数据源及其对应的JPA实体管理器和事务管理器:

在配置类中,我们需要分别为每个数据源创建DataSource、LocalContainerEntityManagerFactoryBean和JpaTransactionManager。这里假设有两个数据源db1和db2:

@Configuration
@EnableTransactionManagement
public class AtomikosConfig {@Bean@Primary@ConfigurationProperties(prefix = "spring.datasource.db1")public DataSource dataSource1() {return new AtomikosDataSourceBean();}@Bean@ConfigurationProperties(prefix = "spring.datasource.db2")public DataSource dataSource2() {return new AtomikosDataSourceBean();}@Bean@Primarypublic LocalContainerEntityManagerFactoryBean entityManagerFactory1() {HibernateJpaVendorAdapter jpaVendorAdapter = new HibernateJpaVendorAdapter();jpaVendorAdapter.setGenerateDdl(true);LocalContainerEntityManagerFactoryBean factory = new LocalContainerEntityManagerFactoryBean();factory.setJpaVendorAdapter(jpaVendorAdapter);factory.setPackagesToScan("com.example.package1");factory.setDataSource(dataSource1());factory.setPersistenceUnitName("persistenceUnit1");return factory;}@Beanpublic LocalContainerEntityManagerFactoryBean entityManagerFactory2() {HibernateJpaVendorAdapter jpaVendorAdapter = new HibernateJpaVendorAdapter();jpaVendorAdapter.setGenerateDdl(true);LocalContainerEntityManagerFactoryBean factory = new LocalContainerEntityManagerFactoryBean();factory.setJpaVendorAdapter(jpaVendorAdapter);factory.setPackagesToScan("com.example.package2");factory.setDataSource(dataSource2());factory.setPersistenceUnitName("persistenceUnit2");return factory;}
}

注意,这里使用了@Primary注解来标记主数据源和对应的实体管理器。

3. Spring Boot+MyBatis集成Atomikos

如果你的项目里同时使用了Spring Boot,MyBatis和Atomikos 。 两个数据源定义两个SqlSessionFactory和两个事务管理器,每个SqlSessionFactory和DataSourceTransactionManager都关联了一个特定的数据源。请注意,在我们定义SqlSessionFactory时,指定了mapper文件的路径,这是必需的,以便MyBatis知道如何将SQL语句映射到你的Java对象。你需要根据你的项目结构来修改这些路径。

@Configuration
@EnableTransactionManagement
public class AtomikosConfig {// 配置第一个数据源@Bean@Primary@ConfigurationProperties(prefix = "spring.datasource.db1")public DataSource dataSource1() {return new AtomikosDataSourceBean();}// 配置第二个数据源@Bean@ConfigurationProperties(prefix = "spring.datasource.db2")public DataSource dataSource2() {return new AtomikosDataSourceBean();}// 配置第一个SqlSessionFactory@Bean@Primarypublic SqlSessionFactory sqlSessionFactory1() throws Exception {SqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();sqlSessionFactoryBean.setDataSource(dataSource1());sqlSessionFactoryBean.setMapperLocations(new PathMatchingResourcePatternResolver().getResources("classpath:mapper/db1/*.xml"));return sqlSessionFactoryBean.getObject();}// 配置第二个SqlSessionFactory@Beanpublic SqlSessionFactory sqlSessionFactory2() throws Exception {SqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();sqlSessionFactoryBean.setDataSource(dataSource2());sqlSessionFactoryBean.setMapperLocations(new PathMatchingResourcePatternResolver().getResources("classpath:mapper/db2/*.xml"));return sqlSessionFactoryBean.getObject();}// 配置第一个事务管理器@Beanpublic DataSourceTransactionManager transactionManager1() {return new DataSourceTransactionManager(dataSource1());}// 配置第二个事务管理器@Beanpublic DataSourceTransactionManager transactionManager2() {return new DataSourceTransactionManager(dataSource2());}
}

4. 在application.properties文件中配置数据源和JPA属性:

spring.datasource.db1.unique-resource-name=datasource1
spring.datasource.db1.xa-data-source-class-name=com.mysql.cj.jdbc.MysqlXADataSource
spring.datasource.db1.xa-properties.databaseName=db1
spring.datasource.db1.xa-properties.url=jdbc:mysql://localhost:3306/db1
spring.datasource.db1.xa-properties.user=root
spring.datasource.db1.xa-properties.password=password
spring.datasource.db1.pool-size=5spring.datasource.db2.unique-resource-name=datasource2
spring.datasource.db2.xa-data-source-class-name=com.mysql.cj.jdbc.MysqlXADataSource
spring.datasource.db2.xa-properties.databaseName=db2
spring.datasource.db2.xa-properties.url=jdbc:mysql://localhost:3306/db2
spring.datasource.db2.xa-properties.user=root
spring.datasource.db2.xa-properties.password=password
spring.datasource.db2.pool-size=5
  1. 创建Atomikos的UserTransaction和TransactionManager实例。

  2. 创建要参与分布式事务的资源(如数据库连接)的XADataSource实例,并将它们注册到Atomikos。

  3. 调用UserTransaction的begin方法开始事务。

  4. 通过XADataSource获取资源连接,进行业务操作。

  5. 调用UserTransaction的commit方法提交事务,或调用rollback方法回滚事务。

4.使用示例

接下来,我们将创建一个简单的服务来演示如何使用在两个数据库上进行分布式事务。这个服务将在两个数据库上进行数据的添加操作。

首先创建两个实体类,分别对应两个数据库的表:

@Entity
@Table(name = "test1")
public class Test1 {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;//...
}@Entity
@Table(name = "test2")
public class Test2 {@Id@GeneratedValue(strategy = GenerationType.IDENTITY)private Long id;//...
}

然后创建对应的Repository:

public interface Test1Repository extends JpaRepository<Test1, Long> {}public interface Test2Repository extends JpaRepository<Test2, Long> {}

接下来创建处理这两个数据库操作的Service:

@Service
public class TestService {private final Test1Repository test1Repository;private final Test2Repository test2Repository;public TestService(Test1Repository test1Repository, Test2Repository test2Repository) {this.test1Repository = test1Repository;this.test2Repository = test2Repository;}@Transactionalpublic void addData() {Test1 test1 = new Test1();//...test1Repository.save(test1);Test2 test2 = new Test2();//...test2Repository.save(test2);}
}

addData方法中,我们在Test1Test2两个表中分别添加数据。由于addData方法添加了@Transactional注解,所以这两个添加数据的操作会在同一个事务中执行。如果在添加数据到Test2表时出现了错误,那么添加数据到Test1表的操作也会被回滚。

在Controller或者其他层调用addData方法,例如:

@RestController
public class TestController {private final TestService testService;public TestController(TestService testService) {this.testService = testService;}@PostMapping("/addData")public ResponseEntity<String> addData() {testService.addData();return ResponseEntity.ok().body("Data added successfully");}
}

5.底层原理

Atomikos 的底层原理是基于 JTA 规范,通过事务管理器协调和管理分布式事务,使用两阶段提交协议保证事务的一致性,提供日志和恢复机制用于持久化和恢复事务状态,以及使用资源适配器与各个资源进行交互。这些组件和机制共同提供了可靠的分布式事务管理功能。

  1. JTA 实现
    Atomikos 实现了 JTA 规范,它提供了 javax.transaction 包中定义的接口和类的实现。这些接口包括 UserTransactionTransactionManagerTransaction 等。Atomikos 利用这些接口和类来进行事务的管理和控制。

  2. 事务管理器(Transaction Manager)
    Atomikos 提供了一个事务管理器,用于协调和管理分布式事务。事务管理器负责事务的创建、启动、提交、回滚和状态管理等。它是 Atomikos 的核心组件,负责处理多个资源参与的分布式事务。

  3. 本地事务管理
    Atomikos 还支持本地事务管理,即仅在单个数据库或资源上执行的事务。对于本地事务,Atomikos 利用底层资源的本地事务管理功能,例如 JDBC 的本地事务或 JMS 的本地事务。

  4. 分布式事务管理
    对于涉及多个资源的分布式事务,Atomikos 使用两阶段提交(Two-Phase Commit,2PC)协议来保证事务的一致性。在这个协议中,事务管理器与各个资源管理器进行通信,确保所有资源都准备好提交事务,并在所有资源都准备好后,进行事务的提交操作。

  5. 日志和恢复
    Atomikos 还提供了日志和恢复机制,用于处理事务的持久化和恢复。它通过将事务的状态和操作记录到日志中,以确保事务的持久性。在系统故障或崩溃后,Atomikos 可以使用日志进行事务的恢复,并保证事务的一致性。

  6. 资源适配器(Resource Adapter)
    Atomikos 使用资源适配器来与各个资源进行交互,例如数据库、消息队列等。资源适配器负责管理资源的连接、事务的参与和操作的执行等。Atomikos 提供了一些内置的资源适配器,同时也支持自定义资源适配器。

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

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

相关文章

gif怎么转换成mp4格式视频

gif怎么转换成mp4格式视频&#xff1f;GIF格式是一种广泛应用的公用图像文件格式标准&#xff0c;具有许多优势。它占用的内存较小&#xff0c;可以实现自动循环播放&#xff0c;并且兼容多个平台。然而&#xff0c;GIF格式也存在一些缺点。例如&#xff0c;它无法处理复杂的图…

C语言深入理解指针(非常详细)(一)

目录 内存和地址内存编址的理解 指针变量和地址取地址操作符&#xff08;&&#xff09;指针变量和解引用操作符&#xff08;*&#xff09;指针变量如何拆解指针类型解引用操作符 指针变量的大小 指针变量类型的意义指针的解引用指针-整数 const修饰指针const修饰变量const修…

day01-ES6新特性以及ReactJS入门

课程介绍 ES6新特性ReactJS入门学习 1、ES6 新特性 1.2、let 和 const 命令 var 之前&#xff0c;我们写js定义变量的时候&#xff0c;只有一个关键字&#xff1a; var var 有一个问题&#xff0c;变量作用域的问题&#xff0c;作用域不可控&#xff0c;就是定义的变量有时会…

Linux操作系统--shell编程(正则表达式)

1..正则表达式概述 正则表达式使用单个字符串来描述、匹配一系列符合某个语法规则的字符串。在很多文本编辑器里,正则表达式通常被用来检索、替换那些符合某个模式的文本。在 Linux 中,grep,sed,awk 等文本处理工具都支持通过正则表达式进行模式匹配。 2.常规的匹配操作 3.…

【踩坑日记】STM32 USART 串口与 FreeRTOS 冲突

文章目录 问题描述问题出现的环境问题解决过程第一步第二步第三步第四步第五步第六步第七步第八步 后续验证一些思考类似的问题后记 问题描述 笔者使用 FreeRTOS 创建了两个任务&#xff0c;使两颗 LED 以不同频率闪烁&#xff0c;但是在加入串口 USART 部分代码后&#xff0c…

java八股文面试[多线程]——指令重排序

关于a的操作&#xff0c;由原来的6个指令&#xff0c;变成了4个指令。 1. 指令重排序的介绍 1&#xff09;指令重排序的类型 在执行程序时为了提高性能&#xff0c;编译器和处理器常常会对指令做重排序。 重排序分三种类型&#xff1a;编译器优化的重排序 编译器在不改变单线…

YOLO V5 和 YOLO V8 对比学习

参考文章&#xff1a; 1、YOLOv5 深度剖析 2、如何看待YOLOv8&#xff0c;YOLOv5作者开源新作&#xff0c;它来了&#xff01;? 3、anchor的简单理解 完整网络结构 YOLO v5和YOLO v8的Head部分 YOLO v8的Head 部分相比 YOLOv5 改动较大&#xff0c;换成了目前主流的解耦头结构…

【springboot】Spring Cache缓存:

文章目录 一、导入Maven依赖&#xff1a;二、实现思路&#xff1a;三、代码开发&#xff1a; 一、导入Maven依赖&#xff1a; <dependency><groupId>org.springframework.boot</groupId><artifactId>spring-boot-starter-cache</artifactId><…

WordPress主题Zing V2.2.1/模块化WordPress响应式通用企业商城主题

WordPress主题Zing V2.2.1&#xff0c;模块化WordPress响应式通用企业商城主题。 功能介绍 百度熊掌号文章实时推送、原创保护 多设备支持自适应布局&#xff0c;支持电脑、Pad、手机以及各种浏览器 SEO优化首页、文章、页面、分类均支持自定义标题、关键字和描述 速度优化…

Django(7)-项目实战-发布会管理

登录功能 模板页面 sign/templates/index.html <!DOCTYPE html> <html> <head><title>Login Page</title> </head> <body><h1>发布会管理</h1><form action"/login/" method"post"><la…

idea 常用插件和常用快捷键 - 记录

idea 常用插件 记得下载插件完成后&#xff0c;点击 Apply 和 OK Alibaba Java Coding Guidelines 作用&#xff1a;使用该插件可以&#xff0c;自动提示相关的语法格式问题&#xff0c;格式参考 阿里巴巴代码规范 详情链接&#xff1a; 代码规范之Alibaba Java Coding G…

EXCEL中点击单元格,所在行和列都改变颜色

在日常工作中&#xff0c;尤其是办公室工作人群&#xff0c;尝尝需要处理大量的数据&#xff0c;在对数据进行修改时&#xff0c;时长发生看错行的事情&#xff0c;导致数据越改越乱&#xff0c;因此&#xff0c;我常用的一种方法就是选中单元格时&#xff0c;所在行、列标记为…

【AI】数学基础——高数(函数微分部分)

参考&#xff1a;https://www.bilibili.com/video/BV1mM411r7ko?p1&vd_source260d5bbbf395fd4a9b3e978c7abde437 唐宇迪&#xff1a;机器学习数学基础 文章目录 1.1 函数1.1.1 函数分类1.1.2 常见函数指/对数函数分段函数原函数&反函数sigmod函数Relu函数(非负函数)复…

keil5 快捷下载STM32系列芯片器件包的方法

以STM32H7系列的器件包为例,官网的下载网址为 https://sadevicepacksprodus.blob.core.windows.net/pack/Keil.STM32H7xx_DFP.3.1.1.pack 其中STM32H7xx为芯片系列编号,3.1.1为器件包的版本 如需下载其他系列和版本的器件包,只需把网址中的编号和版本换成对应的即可(前提是输入…

K8S:K8S自动化运维容器Docker集群

文章目录 一.k8s概述1.k8s是什么2.为什么要用K8S3.作用及功能4.k8s容器集群管理系统 二.K8S的特性1.弹性伸缩2.自我修复3.服务发现和复制均衡4.自动发布和回滚5.集中化配置管理和秘钥管理6.存储编排7.任务批量处理运行 三.K8S的集群架构四.K8S的核心组件1.Master组件&#xff0…

ATA-1222A宽带放大器的电子实验案例(案例合集)

ATA-1222A宽带放大器是安泰电子打造的高带宽功放产品&#xff0c;其采用ClassAB的工作模式&#xff0c;带宽高达22MHz&#xff0c;饱和输出功率40W&#xff0c;能兼容全球不同地区的电源标准要求。凭借其优异的指标参数受到不少电子工程师的喜欢&#xff0c;其在电子实验中的应…

香港服务器快还是台湾服务器快?

​  基于机房位置不同&#xff0c;香港服务器相对于台湾服务器在访问速度方面有一定的优势。香港服务器拥有CN2线路&#xff0c;因此访问速度较快。在网络服务商方面&#xff0c;中华电信等台湾服务商提供的带宽也具有很高的性价比。 香港服务器对大陆用户的影响 对于大陆用户…

WordPress导航主题/酷啦鱼导航主题模板

酷啦鱼导航主题模板&#xff0c;是一款基于WordPress的导航主题&#xff0c;酷啦鱼导航主题是个人基于wordpresscodestar work框架设计的简洁导航主题。 下载地址&#xff1a;https://bbs.csdn.net/topics/616084697

无涯教程-Android Mock Test函数

本节介绍了与 Android 相关的各种模拟测试。您可以在本地计算机上下载这些样本模拟测试,并在方便时离线解决。每个模拟测试均随附一个模拟测试键,可让您验证最终分数并为自己评分。 Mock Test I Mock Test II Mock Test III Mock Test IV Q 1 -什么是Android&#xff1f; A -A…

Python Opencv实践 - Sobel边缘检测

import cv2 as cv import numpy as np import matplotlib.pyplot as pltimg cv.imread("../SampleImages/pomeranian.png", cv.IMREAD_GRAYSCALE) print(img.shape)#Sobel边缘检测 #cv.sobel( src, ddepth, dx, dy[,ksize[, scale[, delta[, borderType]]]] ) #src:…