Java中seata框架的XA模式詳解
XA模式
XA 模式屬于一種強(qiáng)一致性的事務(wù)模式。
前提
支持 XA 模式的數(shù)據(jù)庫。
Java 應(yīng)用通過 JDBC 訪問數(shù)據(jù)庫。
整體機(jī)制
在 Seata 定義的分布式事務(wù)框架內(nèi),利用事務(wù)資源(數(shù)據(jù)庫、消息服務(wù)等)對(duì) XA 協(xié)議提供可回滾、持久化的支持,使用 XA 協(xié)議的機(jī)制來管理分支事務(wù)。

執(zhí)行階段
執(zhí)行 XA Start、業(yè)務(wù) SQL、XA End =》注冊(cè)分支,XA Prepare => 報(bào)告分支事務(wù)的狀態(tài)。
完成階段
執(zhí)行 XA Commit / XA Rollback 操作進(jìn)行分支事務(wù)的提交或者回滾。
XA 模式需要 XAConnection,而獲取 XAConnection 的方式有兩種:
- 方式一、要求開發(fā)者配置 XADataSource。給開發(fā)者增加了認(rèn)知負(fù)擔(dān),需要為 XA 模式專門去學(xué)習(xí)和使用 XA 數(shù)據(jù)源,與透明化 XA 編程模型的設(shè)計(jì)目標(biāo)相悖。
- 方式二、根據(jù)開發(fā)者的普通 DataSource 來創(chuàng)建。對(duì)開發(fā)者比較友好,和 AT 模式一樣,開發(fā)者完全不需要關(guān)心 XA 層面的任何問題,保持本地編程模型即可。
優(yōu)先設(shè)計(jì)實(shí)現(xiàn)第二種方式,數(shù)據(jù)源代理根據(jù)普通數(shù)據(jù)源中獲取的普通 JDBC 連接創(chuàng)建出相應(yīng)的 XAConnection。
類比 AT 模式的數(shù)據(jù)源代理機(jī)制,如下:

但是,第二種方法有局限:無法保證兼容的正確性。
實(shí)際上,這種方法是在做數(shù)據(jù)庫驅(qū)動(dòng)程序要做的事情。不同的廠商、不同版本的數(shù)據(jù)庫驅(qū)動(dòng)實(shí)現(xiàn)機(jī)制是廠商私有的,我們只能保證在充分測(cè)試過的驅(qū)動(dòng)程序上是正確的,開發(fā)者使用的驅(qū)動(dòng)程序版本差異很可能造成機(jī)制的失效。
綜合考慮,XA 模式的數(shù)據(jù)源代理設(shè)計(jì)需要同時(shí)支持第一種方式:基于 XA 數(shù)據(jù)源進(jìn)行代理。
類比 AT 模式的數(shù)據(jù)源代理機(jī)制,如下:

使用方法
每個(gè)服務(wù)的 file.conf、registry.conf 配置文件的配置這里先不提供。
Business服務(wù)
|
|------> Stock服務(wù)
|
|------> Order服務(wù) -----> Account服務(wù)
Business服務(wù)
BusinessService
@GlobalTransactional
public void purchase(String userId, String commodityCode, int orderCount, boolean rollback) {
String xid = RootContext.getXID();
LOGGER.info("New Transaction Begins: " + xid);
String result = stockFeignClient.deduct(commodityCode, orderCount);
if (!SUCCESS.equals(result)) {
throw new RuntimeException("庫存服務(wù)調(diào)用失敗,事務(wù)回滾!");
}
result = orderFeignClient.create(userId, commodityCode, orderCount);
if (!SUCCESS.equals(result)) {
throw new RuntimeException("訂單服務(wù)調(diào)用失敗,事務(wù)回滾!");
}
if (rollback) {
throw new RuntimeException("Force rollback ... ");
}
}BusinessXADataSourceConfiguration
@Configuration
public class BusinessXADataSourceConfiguration {
@Bean
@ConfigurationProperties(prefix = "spring.datasource")
public DruidDataSource dataSource() {
return new DruidDataSource();
}
}Stock服務(wù)
StockBusiness
public void deduct(String commodityCode, int count) {
String xid = RootContext.getXID();
LOGGER.info("deduct stock balance in transaction: " + xid);
jdbcTemplate.update("update seata_stock set count = count - ? where commodity_code = ?",
new Object[] {count, commodityCode});
}application.properties
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver spring.datasource.url=jdbc:mysql://10.211.55.6:3306/seata_stock?useSSL=false&useUnicode=true&characterEncoding=utf-8&allowMultiQueries=true spring.datasource.username=root spring.datasource.password=root
StockXADataSourceConfiguration
@Configuration
public class StockXADataSourceConfiguration {
@Bean
@ConfigurationProperties(prefix = "spring.datasource")
public DruidDataSource druidDataSource() {
return new DruidDataSource();
}
@Bean("dataSourceProxy")
public DataSource dataSource(DruidDataSource druidDataSource) {
return new DataSourceProxyXA(druidDataSource);
}
@Bean("jdbcTemplate")
public JdbcTemplate jdbcTemplate(DataSource dataSourceProxy) {
return new JdbcTemplate(dataSourceProxy);
}
}Order服務(wù)
OrderService
public void create(String userId, String commodityCode, Integer count) {
String xid = RootContext.getXID();
LOGGER.info("create order in transaction: " + xid);
// 定單總價(jià) = 訂購(gòu)數(shù)量(count) * 商品單價(jià)(100)
int orderMoney = count * 100;
// 生成訂單
jdbcTemplate.update("insert seata_order(user_id,commodity_code,count,money) values(?,?,?,?)",
new Object[] {userId, commodityCode, count, orderMoney});
// 調(diào)用賬戶余額扣減
String result = accountFeignClient.reduce(userId, orderMoney);
if (!SUCCESS.equals(result)) {
throw new RuntimeException("Failed to call Account Service. ");
}
}application.properties
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver spring.datasource.url=jdbc:mysql://10.211.55.6:3306/seata_order?useSSL=false&useUnicode=true&characterEncoding=utf-8&allowMultiQueries=true spring.datasource.username=root spring.datasource.password=root
OrderXADataSourceConfiguration
@Configuration
public class OrderXADataSourceConfiguration {
@Bean
@ConfigurationProperties(prefix = "spring.datasource")
public DruidDataSource druidDataSource() {
return new DruidDataSource();
}
@Bean("dataSourceProxy")
public DataSource dataSource(DruidDataSource druidDataSource) {
return new DataSourceProxyXA(druidDataSource);
}
@Bean("jdbcTemplate")
public JdbcTemplate jdbcTemplate(DataSource dataSourceProxy) {
return new JdbcTemplate(dataSourceProxy);
}
}Account服務(wù)
AccountService
@Transactional
public void reduce(String userId, int money) {
String xid = RootContext.getXID();
LOGGER.info("reduce account balance in transaction: " + xid);
jdbcTemplate.update("update seata_account set money = money - ? where user_id = ?", new Object[] {money, userId});
int balance = jdbcTemplate.queryForObject("select money from seata_account where user_id = ?",
new Object[] {userId}, Integer.class);
LOGGER.info("balance after transaction: " + balance);
if (balance < 0) {
throw new RuntimeException("Not Enough Money ...");
}
}application.properties
spring.datasource.driver-class-name=com.mysql.cj.jdbc.Driver spring.datasource.url=jdbc:mysql://10.211.55.6:3306/seata_account?useSSL=false&useUnicode=true&characterEncoding=utf-8&allowMultiQueries=true spring.datasource.username=root spring.datasource.password=root
AccountXADataSourceConfiguration
@Configuration
public class AccountXADataSourceConfiguration {
@Bean
@ConfigurationProperties(prefix = "spring.datasource")
public DruidDataSource druidDataSource() {
return new DruidDataSource();
}
@Bean("dataSourceProxy")
public DataSource dataSource(DruidDataSource druidDataSource) {
return new DataSourceProxyXA(druidDataSource);
}
@Bean("jdbcTemplate")
public JdbcTemplate jdbcTemplate(DataSource dataSourceProxy) {
return new JdbcTemplate(dataSourceProxy);
}
@Bean
public PlatformTransactionManager txManager(DataSource dataSourceProxy) {
return new DataSourceTransactionManager(dataSourceProxy);
}
}啟動(dòng)類需要標(biāo)注 @EnableTransactionManagement 注解。
到此這篇關(guān)于Java中seata框架的XA模式詳解的文章就介紹到這了,更多相關(guān)seata框架的XA模式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Struts2開發(fā)環(huán)境搭建 附簡(jiǎn)單登錄功能實(shí)例
這篇文章主要介紹了Struts2開發(fā)環(huán)境搭建,為大家分享一個(gè)簡(jiǎn)單登錄功能實(shí)例,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11
MyBatisPlus 自定義sql語句的實(shí)現(xiàn)
這篇文章主要介紹了MyBatisPlus 自定義sql語句的實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-08-08
Java和SQL實(shí)現(xiàn)取兩個(gè)字符間的值
這篇文章主要介紹了Java和SQL實(shí)現(xiàn)取兩個(gè)字符間的值操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-06-06

