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

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

		
			com.atomikos
			transactions-jdbc
			4.0.6
		
		
			javax.transaction
			jta
			1.1
		
		
		
			org.springframework
			spring-tx
			${spring-version}
		
	....

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



	
		
			
		
		
			
		
		
	
	
		
			true
		
	
	
		
		
			10
		
	
	
		Oracle数据源
		
			oracle_sources_ds
		
		
			oracle.jdbc.xa.client.OracleXADataSource
		
		
			
				flow
				
				jdbc:oracle:thin:@//192.168.0.101/orcl
			
		
		
			select sysdate from dual
		
		
	
	
		
		
		
			
				jdbc:mysql://localhost:3306/car
				
				root
				
				true
			
		
		
		
		
		
	

发起分布式事务示例代码,这样你可以测试以下两个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配置文件改成:



	
		
		
		
		
	
	
		
		
		
	

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

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

超简单的Atomikos与Spring管理的分布式事务--深入JDBC解释分布式事务的实现_第1张图片

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

超简单的Atomikos与Spring管理的分布式事务--深入JDBC解释分布式事务的实现_第2张图片

 

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----");
	}

}

你可能感兴趣的:(Java程序,spring,分布式)