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
    @Primary
    public 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;
    }

    @Bean
    public 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
    @Primary
    public SqlSessionFactory sqlSessionFactory1() throws Exception {
        SqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();
        sqlSessionFactoryBean.setDataSource(dataSource1());
        sqlSessionFactoryBean.setMapperLocations(new PathMatchingResourcePatternResolver().getResources("classpath:mapper/db1/*.xml"));
        return sqlSessionFactoryBean.getObject();
    }

    // 配置第二个SqlSessionFactory
    @Bean
    public SqlSessionFactory sqlSessionFactory2() throws Exception {
        SqlSessionFactoryBean sqlSessionFactoryBean = new SqlSessionFactoryBean();
        sqlSessionFactoryBean.setDataSource(dataSource2());
        sqlSessionFactoryBean.setMapperLocations(new PathMatchingResourcePatternResolver().getResources("classpath:mapper/db2/*.xml"));
        return sqlSessionFactoryBean.getObject();
    }

    // 配置第一个事务管理器
    @Bean
    public DataSourceTransactionManager transactionManager1() {
        return new DataSourceTransactionManager(dataSource1());
    }

    // 配置第二个事务管理器
    @Bean
    public 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=5

spring.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;
    }

    @Transactional
    public 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 提供了一些内置的资源适配器,同时也支持自定义资源适配器。

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

【C++多线程】C++11互斥锁和条件变量实现生产者消费者模型

先看几个问题&#xff0c;第三个问题可以先看代码然后再理解 Q1&#xff1a;临界区在哪 A1: 队列中元素在「生产者生产&#xff08;push&#xff09;」和「消费者消费&#xff08;pop&#xff09;」时就是临界区 Q2&#xff1a;同步操作在哪 A2: 很显然&#xff0c;队列只有…

pytorch中torch.gather()简单理解

1.作用 从输入张量中按照指定维度进行索引采集操作&#xff0c;返回值是一个新的张量&#xff0c;形状与 index 张量相同&#xff0c;根据指定的索引从输入张量中采集对应的元素。 2.问题 该函数的主要问题主要在dim维度上&#xff0c;dim0 表示沿着第一个维度&#xff08;行…

windows server dfs复制 命名空间

环境准备 1、ad域控服务器 1台 2、文件服务器 2台&#xff0c;要加域 3、windows客户都1台&#xff0c;测试用 实现功能 负载均衡 &#xff08;文件服务器1&#xff1a;负责部门1&#xff0c;部门2的共享文件访问&#xff0c; 文件服务器2&#xff1a;负责部门3&#xff0c;…

【Apollo学习笔记】——规划模块TASK之PIECEWISE_JERK_PATH_OPTIMIZER

文章目录 前言PIECEWISE_JERK_PATH_OPTIMIZER功能简介PIECEWISE_JERK_PATH_OPTIMIZER相关配置PIECEWISE_JERK_PATH_OPTIMIZER总体流程OptimizePathpiecewise_jerk_problem二次规划问题标准形式定义优化变量定义目标函数设计约束OptimizeFormulateProblem计算QP系数矩阵Calculat…

晨控CK-GW208与三菱L系列PLC以TCP通讯手册

晨控CK-GW208是一款支持标准工业以太网协议的IO-LINK主站网关&#xff0c;方便用户快速便捷的集成到 PLC 等控制系统中。 CK-GW208主站网关集成 8 路 IO-LINK 通信端口&#xff0c;采用即插即用模式&#xff0c;无需繁琐的配置&#xff0c;减轻现场安装调试的工作量。为了满足…

Java简便集成工作流(activiti),通用审批系统

前言 activiti工作流引擎项目&#xff0c;企业erp、oa、hr、crm等企事业办公系统轻松落地&#xff0c;请假审批demo从流程绘制到审批结束实例。 一、项目形式 springbootvueactiviti集成了activiti在线编辑器&#xff0c;流行的前后端分离部署开发模式&#xff0c;快速开发平…

机器学习实战之模型的解释性:Scikit-Learn的SHAP和LIME库详解

引言&#xff1a;机器学习模型的“黑箱”困境 机器学习模型的崛起让我们惊叹不已&#xff01;不论是预测房价、识别图片中的猫狗&#xff0c;还是推荐给你喜欢的音乐&#xff0c;这些模型都表现得非常出色。但是&#xff0c;有没有想过&#xff0c;这些模型到底是如何做出这些决…

鸿蒙是一个怎么样的操作系统,真的是安卓套壳吗?

从鸿蒙项目正式推出以来&#xff0c;就一直有各自声音&#xff0c;有看好的&#xff0c;认为鸿蒙的出现将会成为一个智能终端设备操作系统的框架和平台&#xff0c;促进万物互联产业的繁荣发展&#xff1b;也有的人在唱衰&#xff0c;觉得鸿蒙发展不起来&#xff0c;甚至认为鸿…

【计算机基础】一文搞清楚什么是线程/进程/协程

&#x1f449;博主介绍&#xff1a; 博主从事应用安全和大数据领域&#xff0c;有8年研发经验&#xff0c;5年面试官经验&#xff0c;Java技术专家&#xff0c;WEB架构师&#xff0c;阿里云专家博主&#xff0c;华为云云享专家&#xff0c;51CTO 专家博主 ⛪️ 个人社区&#x…

3d Max因卡顿未保存?有什么保护文件和恢复操作呢?

大家在使用3d Max进行建模、渲染和动画制作的过程中&#xff0c;由于各种原因导致软件卡顿或崩溃是很常见的情况。 当卡顿发生时&#xff0c;如果之前的工作没有及时保存&#xff0c;可能会导致数据的丢失和时间的浪费。 一、先来看看保护文件 1、自动保存设置 3d Max提供了自…

机器学习中XGBoost算法调参技巧

本文将详细解释XGBoost中十个最常用超参数的介绍&#xff0c;功能和值范围&#xff0c;及如何使用Optuna进行超参数调优。 对于XGBoost来说&#xff0c;默认的超参数是可以正常运行的&#xff0c;但是如果你想获得最佳的效果&#xff0c;那么就需要自行调整一些超参数来匹配你…

C++——引用

引用的概念 引用不是新定义一个变量&#xff0c;而是给已存在的变量取一个别名&#xff0c;编译器不会因为引用变量而开辟内存空间&#xff0c;它和它引用的变量公用同一块空间。 相当于是给被引用的变量取了一个小名&#xff0c;但是相当于是同一个变量。 类型& 引用变…

ES 7.6 - APi基础操作篇

ES7.6-APi基础操作篇 前言相关知识索引相关创建索引查询索引查询所有索引删除索引关闭与打开索引关闭索引打开索引 冻结与解冻索引冻结索引解冻索引 映射相关创建映射查看映射新增字段映射 文档相关(CURD)新增文档根据ID查询修改文档全量覆盖根据ID选择性修改根据条件批量更新 …

手写数字识别之网络结构

目录 手写数字识别之网络结构 数据处理 经典的全连接神经网络 卷积神经网络 手写数字识别之网络结构 无论是牛顿第二定律任务&#xff0c;还是房价预测任务&#xff0c;输入特征和输出预测值之间的关系均可以使用“直线”刻画&#xff08;使用线性方程来表达&#xff09…

【IMX6ULL驱动开发学习】10.Linux I2C驱动实战:AT24C02驱动设计流程

前情回顾&#xff1a;【IMX6ULL驱动开发学习】09.Linux之I2C框架简介和驱动程序模板_阿龙还在写代码的博客-CSDN博客 目录 一、修改设备树&#xff08;设备树用来指定引脚资源&#xff09; 二、编写驱动 2.1 i2c_drv_read 2.2 i2c_drv_write 2.3 完整驱动程序 三、上机测…

Spring 与【MyBatis 】和【 pageHelper分页插件 】整合

目录 一、Spring整合MyBatis 1. 导入pom依赖 2. 利用mybatis逆向工程生成模型层层代码 3. 编写配置文件 4. 注解式开发 5. 编写Junit测试类 二、AOP整合pageHelper分页插件 1. 创建一个AOP切面 2. Around("execution(* *..*xxx.*xxx(..))") 表达式解析 3. 编…

Visual Studio 2022的MFC框架——WinMain函数

我是荔园微风&#xff0c;作为一名在IT界整整25年的老兵&#xff0c;今天我们来重新审视一下Visual Studio 2022下开发工具的MFC框架知识。 大家还记得创建Win32应用程序是怎么弄的吗&#xff1f; Win32应用程序的建立到运行是有一个个关系分明的步骤的&#xff1a; 1.进入W…

【面试经典150题】删除有序数组中的重复项-JavaScript版

题目链接 思路1&#xff1a;使用set。 /*** param {number[]} nums* return {number}*/ var removeDuplicates function(nums) {const uniqueSetnew Set();for(let i0;i<nums.length;i){uniqueSet.add(nums[i]);}const uniqueArrayArray.from(uniqueSet);nums.length0;nu…

【LeetCode75】第三十五题 统计二叉树中好节点的数目

目录 题目&#xff1a; 示例&#xff1a; 分析&#xff1a; 代码&#xff1a; 题目&#xff1a; 示例&#xff1a; 分析&#xff1a; 给我们一棵二叉树&#xff0c;让我们统计这棵二叉树中好节点的数目。 那么什么是好节点&#xff0c;题目中给出定义&#xff0c;从根节点…

浅谈 Linux 下 vim 的使用

Vim 是从 vi 发展出来的一个文本编辑器&#xff0c;其代码补全、编译及错误跳转等方便编程的功能特别丰富&#xff0c;在程序员中被广泛使用。 Vi 是老式的字处理器&#xff0c;功能虽然已经很齐全了&#xff0c;但还有可以进步的地方。Vim 可以说是程序开发者的一项很好用的工…