超简单的Atomikos与Spring管理的分布式事务--深入JDBC解释分布式事务的实现

maven引入相应的Atomikos依赖类库,除此之外还需要mysql和oracle的jdbc驱动类

		<dependency>
			<groupId>com.atomikos</groupId>
			<artifactId>transactions-jdbc</artifactId>
			<version>4.0.6</version>
		</dependency>
		<dependency>
			<groupId>javax.transaction</groupId>
			<artifactId>jta</artifactId>
			<version>1.1</version>
		</dependency>
		<!-- https://mvnrepository.com/artifact/org.springframework/spring-tx -->
		<dependency>
			<groupId>org.springframework</groupId>
			<artifactId>spring-tx</artifactId>
			<version>${spring-version}</version>
		</dependency>
	....

增加spring配置文件(这里数据库用户密码留空)

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns:aop="http://www.springframework.org/schema/aop" xmlns:context="http://www.springframework.org/schema/context"
	xmlns:tx="http://www.springframework.org/schema/tx"
	xsi:schemaLocation="http://www.springframework.org/schema/beans
         http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
         http://www.springframework.org/schema/aop
         http://www.springframework.org/schema/aop/spring-aop-4.0.xsd
         http://www.springframework.org/schema/context 
		 http://www.springframework.org/schema/context/spring-context-4.0.xsd
		 http://www.springframework.org/schema/tx
		 http://www.springframework.org/schema/tx/spring-tx-4.0.xsd">
	<bean id="springTransactionManager" class="org.springframework.transaction.jta.JtaTransactionManager">
		<property name="transactionManager">
			<ref bean="atomikosTransactionManager" />
		</property>
		<property name="userTransaction">
			<ref bean="atomikosUserTransaction" />
		</property>
		<property name="allowCustomIsolationLevels" value="true" />
	</bean>
	<bean id="atomikosTransactionManager" class="com.atomikos.icatch.jta.UserTransactionManager" init-method="init" destroy-method="close">
		<property name="forceShutdown">
			<value>true</value>
		</property>
	</bean>
	<bean id="atomikosUserTransaction" class="com.atomikos.icatch.jta.UserTransactionImp">
		<!-- 分布式事务超时的时间单位秒 -->
		<property name="transactionTimeout">
			<value>10</value>
		</property>
	</bean>
	<bean id="firstDb" class="com.atomikos.jdbc.AtomikosDataSourceBean" init-method="init" destroy-method="close">
		<description>Oracle数据源</description>
		<property name="uniqueResourceName">
			<value>oracle_sources_ds</value>
		</property>
		<property name="xaDataSourceClassName">
			<value>oracle.jdbc.xa.client.OracleXADataSource</value>
		</property>
		<property name="xaProperties">
			<props>
				<prop key="user">flow</prop>
				<prop key="password"></prop>
				<prop key="uRL">jdbc:oracle:thin:@//192.168.0.101/orcl</prop>
			</props>
		</property>
		<property name="testQuery">
			<value>select sysdate from dual</value>
		</property>
		<property name="poolSize" value="5" />
	</bean>
	<bean id="secondDb" class="com.atomikos.jdbc.AtomikosDataSourceBean" init-method="init" destroy-method="close">
		<property name="uniqueResourceName" value="mysql_sources_ds" />
		<property name="xaDataSourceClassName" value="com.mysql.jdbc.jdbc2.optional.MysqlXADataSource" />
		<property name="xaProperties">
			<props>
				<prop key="url">jdbc:mysql://localhost:3306/car
				</prop>
				<prop key="user">root</prop>
				<prop key="password"></prop>
				<prop key="pinGlobalTxToPhysicalConnection">true</prop>
			</props>
		</property>
		<property name="minPoolSize" value="5" />
		<property name="maxPoolSize" value="10" />
		<property name="borrowConnectionTimeout" value="30" />
		<property name="testQuery" value="select 1" />
	</bean>
</beans>

发起分布式事务示例代码,这样你可以测试以下两个update语句符合ACID的特性,测试的方法可以将第二条Oracle执行的update SQL字符加长,最终结果发现Mysql的update语句没有被提交


public class TestAtomikos {
	public static void main(String[] args) {
		ApplicationContext context = new ClassPathXmlApplicationContext("base_jta.xml");
		PlatformTransactionManager transactionManager = context.getBean(PlatformTransactionManager.class);
		DefaultTransactionDefinition definition = new DefaultTransactionDefinition();
		TransactionStatus transaction = transactionManager.getTransaction(definition);
		DataSource orcl = context.getBean("firstDb", DataSource.class);
		DataSource mysql = context.getBean("secondDb", DataSource.class);
		try (Connection mysqlConn = mysql.getConnection(); Connection orclConn = orcl.getConnection();) {
			Statement mysqlStat = mysqlConn.createStatement();
			String usql = "UPDATE LOGIN_LOG w SET w.USERNAME ='Mysql Name' WHERE w.ID ='a1'";
			mysqlStat.executeUpdate(usql);
			Statement orclStat = orclConn.createStatement();
			String updateSql = "UPDATE LOGIN_LOG w SET w.USERNAME ='Oracle Name' WHERE w.ID ='a2'";
			orclStat.executeUpdate(updateSql);
			transactionManager.commit(transaction);
		} catch (Exception e) {
			e.printStackTrace();
			transactionManager.rollback(transaction);
		}
	}
}

截止上面所说通过第三方类库(Atomikos和Spring全局事务管理器)来实现分布式事务,那么通过JDBC是如何实现,请继续往下看:

首先把spring配置文件改成:

<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
	xmlns:aop="http://www.springframework.org/schema/aop" xmlns:context="http://www.springframework.org/schema/context"
	xmlns:tx="http://www.springframework.org/schema/tx"
	xsi:schemaLocation="http://www.springframework.org/schema/beans
         http://www.springframework.org/schema/beans/spring-beans-4.0.xsd
         http://www.springframework.org/schema/aop
         http://www.springframework.org/schema/aop/spring-aop-4.0.xsd
         http://www.springframework.org/schema/context 
		 http://www.springframework.org/schema/context/spring-context-4.0.xsd
		 http://www.springframework.org/schema/tx
		 http://www.springframework.org/schema/tx/spring-tx-4.0.xsd">
	<bean id="mysql_oracle_ds" class="com.mysql.jdbc.jdbc2.optional.MysqlXADataSource">
		<property name="user" value="root"/>
		<property name="password" value="123456"/>
		<property name="url" value="jdbc:mysql://localhost:3306/car"/>
		<property name="pinGlobalTxToPhysicalConnection" value="true"/>
	</bean>
	<bean id="oracle_sources_ds" class="oracle.jdbc.xa.client.OracleXADataSource">
		<property name="user" value="user"/>
		<property name="password" value="password"/>
		<property name="uRL" value="jdbc:oracle:thin:@//192.168.0.209/orcl"/>
	</bean>
</beans>

这里Spring配置的是XA数据源,在运行前先普及下XA协议

        X/Open XA 接口是双向的系统接口,在事务管理器(Transaction Manager)以及一个或多个
资源管理器(Resource Manager)之间形成通信桥梁。事务管理器控制着JTA 事务,管理事
务生命周期,并协调资源。在JTA 中,事务管理器抽象为javax.transaction.TransactionManager
接口,并通过底层事务服务(即JTS)实现。资源管理器负责控制和管理实际资源(如数据
库或JMS 队列)。下图说明了事务管理器、资源管理器,以及典型JTA 环境中客户端应用之
间的关系:

 而分布式事务中的两阶段提交协议如下图

 

bqual、formatID是可选的。解释如下:

gtrid : 是一个全局事务标识符(global transaction identifier),

bqual:是一个分支限定符(branch qualifier),如果没有提供bqual,那么默认值为空字符串''。

formatID:是一个数字,用于标记gtrid和bqual值的格式,这是一个无符号整数(unsigned integer),也就是说,最小为0。如果没有提供formatID,那么其默认值为1。


class JdbcTest {

	private XADataSource mysqlDS;
	private XADataSource oraDS;

	@BeforeEach
	void setUp() throws Exception {
		ApplicationContext context = new ClassPathXmlApplicationContext("base_jdbc.xml");
		this.mysqlDS = context.getBean("mysql_oracle_ds", XADataSource.class);
		this.oraDS = context.getBean("oracle_sources_ds", XADataSource.class);
	}

	private void output(Object obj) {
		System.out.println(obj);
	}

	@Test
	void testXATransaction() throws Exception {
		XAConnection mysqlXaConn = this.mysqlDS.getXAConnection();
		XAConnection oraXaConn = this.oraDS.getXAConnection();
		XAResource mysqlXaRes = mysqlXaConn.getXAResource();
		XAResource oraXaRes = oraXaConn.getXAResource();
		byte[] gtrid = "g12345".getBytes();
		int formateId = 1;
		byte[] mysqlBqual = "b00001".getBytes();
		Xid mysqlXID = new MysqlXid(gtrid, mysqlBqual, formateId);
		byte[] oraBqual = "b00002".getBytes();
		Xid oraXID = new OracleXid(formateId, gtrid, oraBqual);
		try (Connection mysqlConn = mysqlXaConn.getConnection(); Connection orclConn = oraXaConn.getConnection();) {
			mysqlXaRes.start(mysqlXID, XAResource.TMNOFLAGS);
			Statement mysqlStat = mysqlConn.createStatement();
			String usql = "update LOGIN_LOG set USERNAME=UUID() WHERE ID ='ka==2245555'";
			mysqlStat.executeUpdate(usql);
			mysqlXaRes.end(mysqlXID, XAResource.TMSUCCESS);

			oraXaRes.start(oraXID, XAResource.TMNOFLAGS);
			Statement orclStat = orclConn.createStatement();
			String updateSql = "update  LOGIN_LOG w set w.USERNAME=SYS_GUID() WHERE w.ID ='a123'";
			orclStat.executeUpdate(updateSql);
			oraXaRes.end(oraXID, XAResource.TMSUCCESS);
		} catch (Exception e) {
			e.printStackTrace();
			mysqlXaRes.rollback(mysqlXID);
			oraXaRes.rollback(oraXID);
			return;
		}
		boolean onePhase = false; //TM判断有2个事务分支,所以不能优化为一阶段提交
		try {
			int mysqlPrepare = mysqlXaRes.prepare(mysqlXID);
			int oraPrepare = oraXaRes.prepare(oraXID);
			if (mysqlPrepare == XAResource.XA_OK && oraPrepare == XAResource.XA_OK) {
				mysqlXaRes.commit(mysqlXID, onePhase);
				oraXaRes.commit(oraXID, onePhase);
				assertTrue(true);
			} else {
				mysqlXaRes.rollback(mysqlXID);
				oraXaRes.rollback(oraXID);
				assertTrue(false);
			}
		} catch (Exception e) {
			e.printStackTrace();
			mysqlXaRes.rollback(mysqlXID);
			oraXaRes.rollback(oraXID);
		}
		this.output("finish----");
	}

}

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
第1章 课程介绍 介绍该课程的内容、学习成果、实例,还有学习所需的前提知识。 1-1 导学-分布式事务实践 第2章 事务原则与实现 介绍了事务的四大原则,并通过实例介绍数据库实现事务的方法,以及使用JDBC实现事务的方法。 2-1 事务原则与实现事务 2-2 事务原则与实现:SQL事务 2-3 事务原则与实现JDBC事务(上) 2-4 事务原则与实现JDBC事务(下) 第3章 使用Docker搭建环境 介绍了Docker的使用,通过Docker将课程环境搭建起来,方便那些不了解这些技术的同学之后的学习。 3-1 docker简介与mysql安装-1 3-2 docker简介与mysql安装-2 3-3 SpringBoot基础 第4章 Spring事务机制 介绍了Spring事务机制、事物抽象、内部事务和外部事物,以及常用的几种事务管理实现,包括DataSource、JPA、JMS、JTA都通过实例进行说明。还有XA以及两阶段提交,并通过实例演示了使用JTA,通过两阶段提交,实现多数据源的事务实现。... 4-1 Spring事务机制_基本接口 4-2 Spring事务机制_实现 4-3 Jpa事务实例 4-4 Jms事务原理 4-5 Jms-session事务实例 4-6 Jms-spring事务实例 4-7 外部事务与JTA 4-8 JTA单数据源事务实例 4-9 JTA多数据源事务实例 第5章 分布式系统 介绍了分布式系统的定义、实现原则和几种形式,详细介绍了微服务架构的分布式系统,并使用Spring Cloud框架演示了一个完整的微服务系统的实现过程。 5-1 CAP原则和BASE理论简介 5-2 分布式系统综述 5-3 SpringCloud微服务架构 5-4 实现registry 5-5 实现proxy 5-6 user服务 5-7 order服务 5-8 添加hystrix 5-9 使用feign 5-10 优化服务间调用 第6章 分布式事务实现,模式和技术 介绍分布式事务的定义、原则和实现原则,介绍使用Spring框架实现分布式事务的几种方式,包括使用JTA、Spring事务同步、链式事务等,并通过实战介绍其实现。除此以外还介绍了一些分布式事务相关的技术,如幂等性、全局一致性ID、分布式对象等。... 6-1 分布式事务介绍 6-2 spring分布式事务实现_使用JTA 6-3 spring分布式事务实现_不使用JTA 6-4 实例1-DB-DB 6-5 实例1-DB-DB.链式事务管理器 6-6 实例2-JPA-DB.链式事务管理器 6-7 实例3-JMS-DB.最大努力一次提交 6-8 分布式事务实现模式与技术 6-9 全局一致性ID和分布式对象_ 第7章 分布式事务实现:消息驱动模式 详细介绍3种分布式事务实现的模式中的消息驱动模式并通过完整实例演示了消息驱动模式下,实现微服务系统的分布式事务的完整过程。 7-1 分布式事务实现:消息驱动模式 7-2 消息驱动模式实例:设计 7-3 消息驱动模式实例:创建ticket服务 7-4 消息驱动模式实例:实现基本ticket功能 7-5 消息驱动模式实例:锁票1 7-6 消息驱动模式实例:锁票2 7-7 按消息流程实现业务 7-8 支付过程 7-9 票转移 7-10 错误处理:锁票失败 7-11 错误处理:扣费失败 7-12 并发时的错误处理 第8章 分布式事务实现:Event Sourcing模式 详细介绍了分布式事务实现的模式中的Event Sourcing模式,并通过完整实例演示了Event Sourcing模式下,实现微服务系统的分布式事务的完整过程。 8-1 事件溯源模式介绍 8-2 事件溯源模式与Axon框架-1 8-3 事件溯源模式与Axon框架-2 8-4 使用Axon框架的设计过程介绍 8-5 Axon框架-实例(上) 8-6 Axon框架-实例(下) 8-7 Saga模式和Axon Saga 8-8 聚合命令事件(上) 8-9 聚合命令事件(下) 8-10 实现saga 8-11 实现query 8-12 处理时 8-13 并发测试 8-14 cloud-axon实例:分布式处理介绍 8-15 事件设计 8-16 事件与队列设计 8-17 实现User服务 8-18 实现Ticket服务 8-19 实现Order服务 8-20 实现读写分离 8-21 测试与并发 8-22 事件溯源模式与Axon框架总结 第9章 TCC模式和微服务架构的设计模式 本章介绍TCC模式,也对微服务系统的几种设计模式,以及这些模式下分布式事务实现模式进行了介绍。 9-1 TCC模式介绍 9-2 微服务架构的设计模式 第10章 课程总
编辑推荐: 本文来自于csdn,本篇文章主要介绍了LCN5.0.2有3种模式,分别是LCN模式,TCC模式,TXC模式,希望对您的学习 有所帮助。 一、简介 LCN分布式事务框架其本身并不创建事务,而是基于对本地事务的协调从而达到事务一致性的效果。 LCN模式: LCN模式是通过代理Connection的方式实现对本地事务的操作,然后在由TxManager统一协调控制事务。当本地事务提交回滚或者关闭连接时将会执行假操作,该代理的连接将由LCN连接池管理。 该模式的特点: - 该模式对代码的嵌入性为低。 - 该模式仅限于本地存在连接对象且可通过连接对象控制事务的模块。 - 该模式下的事务提交与回滚是由本地事务方控制,对于数据一致性上有较高的保障。 - 该模式缺陷在于代理的连接需要随事务发起方一共释放连接,增加了连接占用的时间。 TCC模式: TCC事务机制相对于传统事务机制(X/Open XA Two-Phase-Commit),其特征在于它不依赖资源管理器(RM)对XA的支持,而是通过对(由业务系统提供的)业务逻辑的调度来实现分布式事务。主要由三步操作,Try: 尝试执行业务、 Confirm:确认执行业务、 Cancel: 取消执行业务。 该模式的特点: - 该模式对代码的嵌入性高,要求每个业务需要写三种步骤的操作。 - 该模式对有无本地事务控制都可以支持使用面广。 - 数据一致性控制几乎完全由开发者控制,对业务开发难度要求高。 TXC模式: TXC模式命名来源于淘宝,实现原理是在执行SQL之前,先查询SQL的影响数据,然后保存执行的SQL快走信息和创建锁。当需要回滚的时候就采用这些记录数据回滚数据库,目前锁实现依赖redis分布式锁控制。 该模式的特点: - 该模式同样对代码的嵌入性低。 - 该模式仅限于对支持SQL方式的模块支持。 - 该模式由于每次执行SQL之前需要先查询影响数据,因此相比LCN模式消耗资源与时间要多。 - 该模式不会占用数据库的连接资源。 二、原理 核心步骤 1.创建事务组 是指在事务发起方开始执行业务代码之前先调用TxManager创建事务组对象,然后拿到事务标示GroupId的过程。 2.添加事务组 添加事务组是指参与方在执行完业务方法以后,将该模块的事务信息添加通知给TxManager的操作。 3.关闭事务组 是指在发起方执行完业务代码以后,将发起方执行结果状态通知给TxManager的动作。当执行完关闭事务组的方法以后,TxManager将根据事务组信息来通知相应的参与模块提交或回滚事务事务控制原理 LCN事务控制原理是由事务模块TxClient下的代理连接池与TxManager的协调配合完成的事务协调控制。 TxClient的代理连接池实现了javax.sql.DataSource接口,并重写了close方法,事务模块在提交关闭以后TxClient连接池将执行"假关闭"操作,等待TxManager协调完成事务以后在关闭连接。 对于代理连接池的优化 自动时机制,任何通讯都有最大时限制,参与模块在等待通知的状态下也有最大时限制,当过时间限制以后事务模块将先确认事务状态,然后再决定执行提交或者回滚操作,主要为了给最大资源占用时间加上限制。 智能识别创建不同的连接 对于只读操作、非事务操作LCN将不开启代理功能,返回本地连接对象,对于补偿事务的启动方将开启回滚连接对象,执行完业务以后马上回滚事务。 LCN连接重用机制 当模块在同一次事务下被重复执行时,连接资源会被重用,提高连接的使用率。 事务补偿机制 为什么需要事务补偿? 事务补偿是指在执行某个业务方法时,本应该执行成功的操作却因为服务器挂机或者网络抖动等问题导致事务没有正常提交,此种场景就需要通过补偿来完成事务,从而达到事务的一致性。 补偿机制的触发条件? 当执行关闭事务组步骤时,若发起方接受到失败的状态后将会把该次事务识别为待补偿事务,然后发起方将该次事务数据异步通知给TxManager。TxManager接受到补偿事务以后先通知补偿回调地址,然后再根据是否开启自动补偿事务状态来补偿或保存该次切面事务数据。 补偿事务机制 LCN的补偿事务原理是模拟上次失败事务的请求,然后传递给TxClient模块然后再次执行该次请求事务。 模拟场景演示 若存在事务发起方、参与方A、参与方B。调用关系图如下 那么他们正常执行业务的时序图为: 若参与方B出现异常,那么他们的业务时序图为: 若他们的调用关系是这样的情况 此时发生参与方B出现异常时他们的时序图为: 三、使用 环境: SpringBoot 2.0.
Boot分布式事务的解决方案是JTA,即Java Transaction API。Spring Boot官方提供了Atomikos的解决思路。在项目中引入Atomikos的依赖,然后在配置文件中配置数据源和事务管理器即可实现分布式事务。具体步骤如下: 1.在pom.xml文件中引入Atomikos的依赖: ```xml <dependency> <groupId>com.atomikos</groupId> <artifactId>transactions-jta</artifactId> <version>4.0.6</version> </dependency> ``` 2.在application.properties文件中配置数据源和事务管理器: ```properties spring.datasource.url=jdbc:mysql://localhost:3306/test spring.datasource.username=root spring.datasource.password=root spring.datasource.driver-class-name=com.mysql.jdbc.Driver spring.jta.atomikos.datasource.ds1.unique-resource-name=ds1 spring.jta.atomikos.datasource.ds1.xa-data-source-class-name=com.mysql.jdbc.jdbc2.optional.MysqlXADataSource spring.jta.atomikos.datasource.ds1.xa-properties.url=jdbc:mysql://localhost:3306/test spring.jta.atomikos.datasource.ds1.xa-properties.user=root spring.jta.atomikos.datasource.ds1.xa-properties.password=root spring.jta.atomikos.transaction-manager-id=tm ``` 3.在代码中使用@Transactional注解开启事务: ```java @Service public class UserServiceImpl implements UserService { @Autowired private UserDao userDao; @Transactional @Override public void transfer(int fromUserId, int toUserId, double amount) { userDao.decreaseBalance(fromUserId, amount); userDao.increaseBalance(toUserId, amount); } } ``` 以上就是Spring Boot分布式事务的解决方案。需要注意的是,使用JTA需要在应用服务器中配置JTA事务管理器,例如Tomcat需要配置Bitronix或Atomikos。同时,JTA也有一些缺点,例如性能较差,配置较为复杂等。因此,在实际项目中需要根据具体情况选择合适的事务解决方案。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

victorkevin

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值