SpringBoot2整合JTA組件實現(xiàn)多數(shù)據(jù)源事務管理
一、JTA組件簡介
1、JTA基本概念
JTA即Java-Transaction-API,JTA允許應用程序執(zhí)行分布式事務處理,即在兩個或多個網(wǎng)絡計算機資源上訪問并且更新數(shù)據(jù)。JDBC驅動程序對JTA的支持極大地增強了數(shù)據(jù)訪問能力。
XA協(xié)議是數(shù)據(jù)庫層面的一套分布式事務管理的規(guī)范,JTA是XA協(xié)議在Java中的實現(xiàn),多個數(shù)據(jù)庫或是消息廠商實現(xiàn)JTA接口,開發(fā)人員只需要調用SpringJTA接口即可實現(xiàn)JTA事務管理功能。
JTA事務比JDBC事務更強大。一個JTA事務可以有多個參與者,而一個JDBC事務則被限定在一個單一的數(shù)據(jù)庫連接。下列任一個Java平臺的組件都可以參與到一個JTA事務中
2、分布式事務
分布式事務(DistributedTransaction)包括事務管理器(TransactionManager)和一個或多個支持 XA 協(xié)議的資源管理器 ( Resource Manager )。
資源管理器是任意類型的持久化數(shù)據(jù)存儲容器,例如在開發(fā)中常用的關系型數(shù)據(jù)庫:MySQL,Oracle等,消息中間件RocketMQ、RabbitMQ等。
事務管理器提供事務聲明,事務資源管理,同步,事務上下文傳播等功能,并且負責著所有事務參與單元者的相互通訊的責任。JTA規(guī)范定義了事務管理器與其他事務參與者交互的接口,其他的事務參與者與事務管理器進行交互。
二、SpringBoot整合JTA
項目整體結構圖
1、核心依賴
<!--SpringBoot核心依賴--> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-web</artifactId> </dependency> <!--JTA組件核心依賴--> <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-starter-jta-atomikos</artifactId> </dependency>
2、環(huán)境配置
這里jtaManager的配置,在日志輸出中非常關鍵。
spring: jta: transaction-manager-id: jtaManager # 數(shù)據(jù)源配置 datasource: type: com.alibaba.druid.pool.DruidDataSource data01: driverClassName: com.mysql.jdbc.Driver dbUrl: jdbc:mysql://localhost:3306/data-one username: root password: 000000 data02: driverClassName: com.mysql.jdbc.Driver dbUrl: jdbc:mysql://localhost:3306/data-two username: root password: 000000
3、核心容器
這里兩個數(shù)據(jù)庫連接的配置手法都是一樣的,可以在源碼中自行下載閱讀?;舅悸范际前褦?shù)據(jù)源交給JTA組件來統(tǒng)一管理,方便事務的通信。
數(shù)據(jù)源參數(shù)
@Component @ConfigurationProperties(prefix = "spring.datasource.data01") public class DruidOneParam { private String dbUrl; private String username; private String password; private String driverClassName; }
JTA組件配置
package com.jta.source.conifg; @Configuration @MapperScan(basePackages = {"com.jta.source.mapper.one"},sqlSessionTemplateRef = "data01SqlSessionTemplate") public class DruidOneConfig { private static final Logger LOGGER = LoggerFactory.getLogger(DruidOneConfig.class) ; @Resource private DruidOneParam druidOneParam ; @Primary @Bean("dataSourceOne") public DataSource dataSourceOne () { // 設置數(shù)據(jù)庫連接 MysqlXADataSource mysqlXADataSource = new MysqlXADataSource(); mysqlXADataSource.setUrl(druidOneParam.getDbUrl()); mysqlXADataSource.setUser(druidOneParam.getUsername()); mysqlXADataSource.setPassword(druidOneParam.getPassword()); mysqlXADataSource.setPinGlobalTxToPhysicalConnection(true); // 事務管理器 AtomikosDataSourceBean atomikosDataSourceBean = new AtomikosDataSourceBean(); atomikosDataSourceBean.setXaDataSource(mysqlXADataSource); atomikosDataSourceBean.setUniqueResourceName("dataSourceOne"); return atomikosDataSourceBean; } @Primary @Bean(name = "sqlSessionFactoryOne") public SqlSessionFactory sqlSessionFactoryOne( @Qualifier("dataSourceOne") DataSource dataSourceOne) throws Exception{ // 配置Session工廠 SqlSessionFactoryBean sessionFactory = new SqlSessionFactoryBean(); sessionFactory.setDataSource(dataSourceOne); ResourcePatternResolver resolver = new PathMatchingResourcePatternResolver(); sessionFactory.setMapperLocations(resolver.getResources("classpath*:/dataOneMapper/*.xml")); return sessionFactory.getObject(); } @Primary @Bean(name = "data01SqlSessionTemplate") public SqlSessionTemplate sqlSessionTemplate( @Qualifier("sqlSessionFactoryOne") SqlSessionFactory sqlSessionFactory) { // 配置Session模板 return new SqlSessionTemplate(sqlSessionFactory); } }
4、測試對比
這里通過兩個方法測試結果做對比,在兩個數(shù)據(jù)源之間進行數(shù)據(jù)操作時,只需要在接口方法加上@Transactional注解即可,這樣保證數(shù)據(jù)在兩個數(shù)據(jù)源間也可以保證一致性。
@Service public class TransferServiceImpl implements TransferService { @Resource private UserAccount01Mapper userAccount01Mapper ; @Resource private UserAccount02Mapper userAccount02Mapper ; @Override public void transfer01() { userAccount01Mapper.transfer("jack",100); System.out.println("i="+1/0); userAccount02Mapper.transfer("tom",100); } @Transactional @Override public void transfer02() { userAccount01Mapper.transfer("jack",200); System.out.println("i="+1/0); userAccount02Mapper.transfer("tom",200); } }
三、JTA組件小結
在上面JTA實現(xiàn)多數(shù)據(jù)源的事務管理,使用方式還是相對簡單,通過兩階段的提交,可以同時管理多個數(shù)據(jù)源的事務。但是暴露出的問題也非常明顯,就是比較嚴重的性能問題,由于同時操作多個數(shù)據(jù)源,如果其中一個數(shù)據(jù)源獲取數(shù)據(jù)的時間過長,會導致整個請求都非常的長,事務時間太長,鎖數(shù)據(jù)的時間就會太長,自然就會導致低性能和低吞吐量。
因此在實際開發(fā)過程中,對性能要求比較高的系統(tǒng)很少使用JTA組件做事務管理。作為一個輕量級的分布式事務解決方案,在小的系統(tǒng)中還是值得推薦嘗試的。
最后作為Java下的API,原理和用法還是值得學習一下,開闊眼界和思路。
四、源代碼地址
GitHub·地址
https://github.com/cicadasmile/middle-ware-parent
GitEE·地址
https://gitee.com/cicadasmile/middle-ware-parent
到此這篇關于SpringBoot2整合JTA組件實現(xiàn)多數(shù)據(jù)源事務管理的文章就介紹到這了,更多相關SpringBoot2 JTA多數(shù)據(jù)源事務管理內容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!
相關文章
SpringBoot JPA懶加載失效的解決方案(親測有效)
這篇文章主要介紹了SpringBoot JPA懶加載失效的解決方案(親測有效),具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教2021-08-08Springboot與vue實現(xiàn)文件導入方法具體介紹
文件導入時大多數(shù)項目無法回避的問題,這兩天深入學習了文件導入,在這里進行記錄,使用到的技術是Springboot+Vue,前端組件使用el-upload2023-02-02