分布式事务模型DTP

2023-10-29 18:08
文章标签 模型 分布式 事务 dtp

本文主要是介绍分布式事务模型DTP,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

   X/Open DTP(X/Open Distributed Transaction Processing Reference Model) 是X/Open 这个组织定义的一套分布式事务的标准,也就是了定义了规范和API接口,由厂商进行具体的实现

    X/Open DTP 定义了三个组件: AP,TM,RM
AP(Application Program):也就是应用程序,可以理解为使用DTP的程序
RM(Resource Manager):资源管理器,这里可以理解为一个DBMS系统,或者消息服务器管理系统,应用程序通过资源管理器对资源进行控制。资源必须实现XA定义的接口
TM(Transaction Manager):事务管理器,负责协调和管理事务,提供给AP应用程序编程接口(TX协议)以及管理资源管理器
其中,AP 可以和TM 以及 RM 通信,TM 和 RM 互相之间可以通信,DTP模型里面定义了XA接口,TM 和 RM 通过XA接口进行双向通信,例如:TM通知RM提交事务或者回滚事务,RM把提交结果通知给TM。AP和RM之间则通过RM提供的Native API 进行资源控制,这个没有进行约API和规范,各个厂商自己实现自己的资源控制,比如Oracle自己的数据库驱动程序。


其中在DTP定了以下几个概念:


事务:一个事务是一个完整的工作单元,由多个独立的计算任务组成,这多个任务在逻辑上是原子的。
全局事务:对于一次性操作多个资源管理器的事务,就是全局事务
分支事务:在全局事务中,某一个资源管理器有自己独立的任务,这些任务的集合作为这个资源管理器的分支任务
控制线程:用来表示一个工作线程,主要是关联AP,TM,RM三者的一个线程,也就是事务上下文环境。简单的说,就是需要标识一个全局事务以及分支事务的关系。

分布式事务模型 使用二阶段提交协议2PC(Two-phaseCommit) 实现多数据源事务处理

准备阶段
事务协调者(事务管理器)给每个参与者(资源管理器)发送Prepare消息,每个参与者要么直接返回失败(如权限验证失败),要么在本地执行事务,写本地的redo和undo日志,但不提交,到达一种“万事俱备,只欠东风”的状态。

提交阶段
如果协调者收到了参与者的失败消息或者超时,直接给每个参与者发送回滚(Rollback)消息;否则,发送提交(Commit)消息;参与者根据协调者的指令执行提交或者回滚操作,释放所有事务处理过程中使用的锁资源。(注意:必须在最后阶段释放锁资源)


出现部分资源失败后的处理情况

二阶段提交看起来确实能够提供原子性的操作,但是不幸的事,二阶段提交还是有几个缺点的、

1、同步阻塞问题。执行过程中,所有参与节点都是事务阻塞型的。当参与者占有公共资源时,其他第三方节点访问公共资源不得不处于阻塞状态。
2、单点故障。由于协调者的重要性,一旦协调者发生故障。参与者会一直阻塞下去。尤其在第二阶段,协调者发生故障,那么所有的参与者还都处于锁定事务资源的状态中,而无法继续完成事务操作。(如果是协调者挂掉,可以重新选举一个协调者,但是无法解决因为协调者宕机导致的参与者处于阻塞状态的问题)
3、数据不一致。在二阶段提交的阶段二中,当协调者向参与者发送commit请求之后,发生了局部网络异常或者在发送commit请求过程中协调者发生了故障,这回导致只有一部分参与者接受到了commit请求。而在这部分参与者接到commit请求之后就会执行commit操作。但是其他部分未接到commit请求的机器则无法执行事务提交。于是整个分布式系统便出现了数据部一致性的现象。
4、二阶段无法解决的问题:协调者再发出commit消息之后宕机,而唯一接收到这条消息的参与者同时也宕机了。那么即使协调者通过选举协议产生了新的协调者,这条事务的状态也是不确定的,没人知道事务是否被已经提交。

二。DTP编程模型JTA

  JTA(Java Transaction API)是符合X/Open DTP的一个编程模型,事务管理和资源管理器支架也是用了XA协议。

1》TA的构成
a、高层应用事务界定接口,供事务客户界定事务边界的
b、X/Open XA协议(资源之间的一种标准化的接口)的标准Java映射,它可以使事务性的资源管理器参与由外部事务管理器控制的事务中
c、高层事务管理器接口,允许应用程序服务器为其管理的应用程序界定事务的边界 

2》JTA的主要接口 
位于javax.transaction包中
a、UserTransaction接口:让应用程序得以控制事务的开始、挂起、提交、回滚等。由Java客户端程序或EJB调用。
b、TransactionManager 接口:用于应用服务器管理事务状态
c、Transaction接口:用于执行相关事务操作
d、XAResource接口:用于在分布式事务环境下,协调事务管理器和资源管理器的工作
e、Xid接口:为事务标识符的Java映射
注:前3个接口位于Java EE版的类库 javaee.jar 中,Java SE中没有提供!UserTransaction是编程常用的接口
注意的是JTA只提供了接口,没有具体的实现。
JTS(Java Transaction Service)是服务OTS的JTA的实现。简单的说JTS实现了JTA接口,并且符合OTS的规范。
JTA的事务周期可横跨多个JDBC Connection生命周期,对众多Connection进行调度,实现其事务性要求。
JTA可以处理任何提供符合XA接口的资源。包括:JDBC连接,数据库,JMS,商业对象等等。
3》JTA实现

JOTM(Java Open Transaction Manager)是ObjectWeb的一个开源JTA实现,它本身也是开源应用程序服务器JOnAS(Java Open Application Server)的一部分,为其提供JTA分布式事务的功能。官网:http://jotm.objectweb.org

Atomikos 是一个为Java平台提供增值服务的并且开源类事务管理器提供 以下是包括在这个开源版本中的一些功能:
1 全面崩溃 / 重启恢复
2 兼容标准的SUN公司JTA API
3 嵌套事务
4 为XA和非XA提供内置的JDBC适配器
官网:https://www.atomikos.com/

三。Spring集成JTA

因为spring的
<tx:annotation-driven transaction-manager="springTransactionManager"/>  只能指定一个事务管理器 所以多数据源下 肯定不能使用同一个事务管理器
必须使用jta事务

1》使用jtom配置jta实现
模拟环境 
  mysql 下存在表 mymoney 存在用户zs 余额 1000 建表sql

[html]  view plain copy
  1. CREATE DATABASE IF NOT EXISTS dtp;  
  2. USE dtp;  
  3. CREATE TABLE mymoney(  
  4.   id INT PRIMARY KEY AUTO_INCREMENT,  
  5.   NAME VARCHAR(20),  
  6.   lostedmoney INT  
  7. );  
  8. INSERT INTO mymoney(id,NAME,lostedmoney) VALUES(1,'zs',1000);  
  oracle下存在表mymoney 存在用户ls 余额300 建表sql
[html]  view plain copy
  1. oracle下 scott账号下 创建相同表  
  2. CREATE TABLE mymoney(  
  3.   id number PRIMARY KEY ,  
  4.   NAME VARCHAR2(20),  
  5.   lostedmoney number  
  6. );  
  7. INSERT INTO mymoney(id,NAME,lostedmoney) VALUES(2,'ls',300);  
模拟zs给ls转账 自然涉及到多数据源分布式事务
配置spring支持jtom的环境 (使用xapool管理数据源)
maven依赖:

[html]  view plain copy
  1. <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  2.     xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">  
  3.     <modelVersion>4.0.0</modelVersion>  
  4.     <groupId>cn.et</groupId>  
  5.     <artifactId>spring</artifactId>  
  6.     <version>0.0.1-SNAPSHOT</version>  
  7.     <dependencies>  
  8.         <dependency>  
  9.             <groupId>org.springframework</groupId>  
  10.             <artifactId>spring-context</artifactId>  
  11.             <version>4.3.13.RELEASE</version>  
  12.         </dependency>  
  13.         <dependency>  
  14.             <groupId>org.springframework</groupId>  
  15.             <artifactId>spring-tx</artifactId>  
  16.             <version>4.3.13.RELEASE</version>  
  17.         </dependency>  
  18.         <dependency>  
  19.             <groupId>org.springframework</groupId>  
  20.             <artifactId>spring-aop</artifactId>  
  21.             <version>4.3.13.RELEASE</version>  
  22.         </dependency>  
  23.         <dependency>  
  24.             <groupId>org.springframework</groupId>  
  25.             <artifactId>spring-jdbc</artifactId>  
  26.             <version>4.3.13.RELEASE</version>  
  27.         </dependency>  
  28.         <!-- https://mvnrepository.com/artifact/jotm/jotm -->  
  29.         <dependency>  
  30.             <groupId>jotm</groupId>  
  31.             <artifactId>jotm</artifactId>  
  32.             <version>2.0.10</version>  
  33.         </dependency>  
  34.         <!-- https://mvnrepository.com/artifact/org.ow2.jotm/jotm-core -->  
  35.         <dependency>  
  36.             <groupId>org.ow2.jotm</groupId>  
  37.             <artifactId>jotm-core</artifactId>  
  38.             <version>2.2.3</version>  
  39.         </dependency>  
  40.         <!-- https://mvnrepository.com/artifact/org.ow2.jotm/jotm-datasource -->  
  41.         <dependency>  
  42.             <groupId>org.ow2.jotm</groupId>  
  43.             <artifactId>jotm-datasource</artifactId>  
  44.             <version>2.2.3</version>  
  45.         </dependency>  
  46.         <!-- https://mvnrepository.com/artifact/org.ow2.jotm/jotm-standalone -->  
  47.         <dependency>  
  48.             <groupId>org.ow2.jotm</groupId>  
  49.             <artifactId>jotm-standalone</artifactId>  
  50.             <version>2.2.3</version>  
  51.         </dependency>  
  52.         <!-- https://mvnrepository.com/artifact/javax.resource/javax.resource-api -->  
  53.         <dependency>  
  54.             <groupId>javax.resource</groupId>  
  55.             <artifactId>javax.resource-api</artifactId>  
  56.             <version>1.7</version>  
  57.         </dependency>  
  58.         <!-- https://mvnrepository.com/artifact/com.experlog/xapool -->  
  59.         <dependency>  
  60.             <groupId>com.experlog</groupId>  
  61.             <artifactId>xapool</artifactId>  
  62.             <version>1.5.0</version>  
  63.         </dependency>  
  64.         <!-- https://mvnrepository.com/artifact/mysql/mysql-connector-java -->  
  65.         <dependency>  
  66.             <groupId>mysql</groupId>  
  67.             <artifactId>mysql-connector-java</artifactId>  
  68.             <version>5.1.44</version>  
  69.         </dependency>  
  70.         <dependency>  
  71.             <groupId>com.jslsolucoes</groupId>  
  72.             <artifactId>ojdbc6</artifactId>  
  73.             <version>11.2.0.1.0</version>  
  74.         </dependency>  
  75.     </dependencies>  
  76. </project>  
添加spring配置文件 配置 两个数据源 并且将两个支持xa的数据源 绑定到一个资源管理器 UserTranscation中
[html]  view plain copy
  1. <?xml version="1.0" encoding="UTF-8"?>    
  2. <beans xmlns="http://www.springframework.org/schema/beans"    
  3.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"   
  4.     xmlns:aop="http://www.springframework.org/schema/aop"    
  5.     xmlns:tx="http://www.springframework.org/schema/tx"  
  6.     xmlns:context="http://www.springframework.org/schema/context"    
  7.     xsi:schemaLocation="      
  8.         http://www.springframework.org/schema/beans       
  9.         http://www.springframework.org/schema/beans/spring-beans-4.0.xsd      
  10.         http://www.springframework.org/schema/tx       
  11.         http://www.springframework.org/schema/tx/spring-tx-4.0.xsd      
  12.         http://www.springframework.org/schema/aop    
  13.         http://www.springframework.org/schema/aop/spring-aop-4.0.xsd      
  14.         http://www.springframework.org/schema/context       
  15.         http://www.springframework.org/schema/context/spring-context-4.0.xsd      
  16. ">    
  17.   <context:component-scan base-package="cn"></context:component-scan>  
  18.   <bean id="txCurrent" class="org.objectweb.jotm.Current"></bean>  
  19.    
  20.   <!-- 使用xapool配置支持xa的资源(mysql数据库)连接池 -->  
  21.   <bean id="xaMysqlDataSource" class="org.enhydra.jdbc.pool.StandardXAPoolDataSource" destroy-method="shutdown">  
  22.      <property name="dataSource">  
  23.          <!-- 定义支持mysql的xa数据源 -->  
  24.           <bean class="org.enhydra.jdbc.standard.StandardXADataSource" destroy-method="shutdown">  
  25.             <property name="driverName" value="com.mysql.jdbc.Driver"></property>  
  26.             <property name="url" value="jdbc:mysql://localhost:3306/dtp"></property>  
  27.             <!-- 注意用户名密码不能配在这里 否则无法读取 -->  
  28.             <!-- 注册到txCurrent -->  
  29.             <property name="transactionManager" ref="txCurrent"></property>  
  30.           </bean>  
  31.              
  32.      </property>  
  33.      <property name="user" value="root"></property>  
  34.      <property name="password" value="123456"></property>  
  35.   </bean>  
  36.    <bean id="mysqlJdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">    
  37.         <property name="dataSource" ref="xaMysqlDataSource" />    
  38.    </bean>    
  39.      
  40.    <!-- 使用xapool配置支持xa的资源(oracle数据库)连接池 -->  
  41.    <bean id="xaOracleDataSource" class="org.enhydra.jdbc.pool.StandardXAPoolDataSource" destroy-method="shutdown">  
  42.      <property name="dataSource">  
  43.          <!-- 定义支持mysql的xa数据源 -->  
  44.           <bean class="org.enhydra.jdbc.standard.StandardXADataSource" destroy-method="shutdown">  
  45.             <property name="driverName" value="oracle.jdbc.OracleDriver"></property>  
  46.             <property name="url" value="jdbc:oracle:thin:@localhost:1521:orcl"></property>  
  47.             <!-- 注册到txCurrent -->  
  48.             <property name="transactionManager" ref="txCurrent"></property>  
  49.           </bean>  
  50.      </property>  
  51.      <property name="user" value="scott"></property>  
  52.      <property name="password" value="tiger"></property>  
  53.   </bean>  
  54.    <bean id="oracleJdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">    
  55.         <property name="dataSource" ref="xaOracleDataSource" />    
  56.    </bean>    
  57.      
  58.      
  59.    <!-- 配置jta事务  userTransaction必须将所有的数据源(RM)注册到txCurrent中-->  
  60.    <bean id="springTransactionManager" class="org.springframework.transaction.jta.JtaTransactionManager">    
  61.         <property name="userTransaction" ref="txCurrent" />    
  62.    </bean>    
  63.    <tx:annotation-driven transaction-manager="springTransactionManager"/>    
  64. </beans>  
添加zs的数据处理类

[html]  view plain copy
  1. package cn.et.dao;  
  2.   
  3. import javax.annotation.Resource;  
  4. import org.springframework.beans.factory.annotation.Autowired;  
  5. import org.springframework.jdbc.core.JdbcTemplate;  
  6. import org.springframework.stereotype.Repository;  
  7. import org.springframework.transaction.annotation.Transactional;  
  8. @Transactional  
  9. @Repository  
  10. public class UserZsDao {  
  11.     @Autowired  
  12.     private UserLsDao userLsDao;  
  13.     @Resource(name="mysqlJdbcTemplate")  
  14.     JdbcTemplate mysqlJdbcTemplate;  
  15.     public void zsMinus(int money){  
  16.         //zs的mysql扣款  
  17.         String sql="update MYMONEY set lostedmoney=lostedmoney-"+money+" where id=1";  
  18.         mysqlJdbcTemplate.execute(sql);  
  19.         //调用ls的oracle加钱  
  20.         userLsDao.lsAdd(money);  
  21.         //模拟出现异常 查看数据两个操作数据是否都回滚  
  22.         String a=null;  
  23.         a.toCharArray();  
  24.     }  
  25. }  
处理ls的数据处理类

[html]  view plain copy
  1. package cn.et.dao;  
  2.   
  3. import javax.annotation.Resource;  
  4.   
  5. import org.springframework.jdbc.core.JdbcTemplate;  
  6. import org.springframework.stereotype.Repository;  
  7. import org.springframework.transaction.annotation.Transactional;  
  8. @Transactional  
  9. @Repository  
  10. public class UserLsDao {  
  11.       
  12.     @Resource(name="oracleJdbcTemplate")  
  13.     JdbcTemplate oracleJdbcTemplate;  
  14.       
  15.     public void lsAdd(int money){  
  16.         String sql="update MYMONEY set lostedmoney=lostedmoney+"+money+" where id=2";  
  17.         oracleJdbcTemplate.execute(sql);  
  18.     }  
  19. }  
添加测试类
[html]  view plain copy
  1. package cn.et.dao;  
  2.   
  3. import org.springframework.context.support.ClassPathXmlApplicationContext;  
  4.   
  5. public class Test {  
  6.   
  7.     public static void main(String[] args) {  
  8.         ClassPathXmlApplicationContext cpxac=new ClassPathXmlApplicationContext("spring.xml");  
  9.         UserZsDao userZsDao=(UserZsDao)cpxac.getBean("userZsDao");  
  10.         userZsDao.zsMinus(100);  
  11.         cpxac.close();  
  12.     }  
  13.   
  14. }  
运行后 发现出现异常正常回滚

这篇关于分布式事务模型DTP的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/302404

相关文章

详解如何使用Python从零开始构建文本统计模型

《详解如何使用Python从零开始构建文本统计模型》在自然语言处理领域,词汇表构建是文本预处理的关键环节,本文通过Python代码实践,演示如何从原始文本中提取多尺度特征,并通过动态调整机制构建更精确... 目录一、项目背景与核心思想二、核心代码解析1. 数据加载与预处理2. 多尺度字符统计3. 统计结果可

SpringBoot整合Sa-Token实现RBAC权限模型的过程解析

《SpringBoot整合Sa-Token实现RBAC权限模型的过程解析》:本文主要介绍SpringBoot整合Sa-Token实现RBAC权限模型的过程解析,本文给大家介绍的非常详细,对大家的学... 目录前言一、基础概念1.1 RBAC模型核心概念1.2 Sa-Token核心功能1.3 环境准备二、表结

MySQL 事务的概念及ACID属性和使用详解

《MySQL事务的概念及ACID属性和使用详解》MySQL通过多线程实现存储工作,因此在并发访问场景中,事务确保了数据操作的一致性和可靠性,下面通过本文给大家介绍MySQL事务的概念及ACID属性和... 目录一、什么是事务二、事务的属性及使用2.1 事务的 ACID 属性2.2 为什么存在事务2.3 事务

Golang实现Redis分布式锁(Lua脚本+可重入+自动续期)

《Golang实现Redis分布式锁(Lua脚本+可重入+自动续期)》本文主要介绍了Golang分布式锁实现,采用Redis+Lua脚本确保原子性,持可重入和自动续期,用于防止超卖及重复下单,具有一定... 目录1 概念应用场景分布式锁必备特性2 思路分析宕机与过期防止误删keyLua保证原子性可重入锁自动

基于MongoDB实现文件的分布式存储

《基于MongoDB实现文件的分布式存储》分布式文件存储的方案有很多,今天分享一个基于mongodb数据库来实现文件的存储,mongodb支持分布式部署,以此来实现文件的分布式存储,需要的朋友可以参考... 目录一、引言二、GridFS 原理剖析三、Spring Boot 集成 GridFS3.1 添加依赖

Spring Boot 事务详解(事务传播行为、事务属性)

《SpringBoot事务详解(事务传播行为、事务属性)》SpringBoot提供了强大的事务管理功能,通过@Transactional注解可以方便地配置事务的传播行为和属性,本文将详细介绍Spr... 目录Spring Boot 事务详解引言声明式事务管理示例编程式事务管理示例事务传播行为1. REQUI

MySQL中的事务隔离级别详解

《MySQL中的事务隔离级别详解》在MySQL中,事务(Transaction)是一个执行单元,它要么完全执行,要么完全回滚,以保证数据的完整性和一致性,下面给大家介绍MySQL中的事务隔离级别详解,... 目录一、事务并发问题二、mysql 事务隔离级别1. READ UNCOMMITTED(读未提交)2

如何合理使用Spring的事务方式

《如何合理使用Spring的事务方式》:本文主要介绍如何合理使用Spring的事务方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、介绍1.1、底层构造1.1.事务管理器1.2.事务定义信息1.3.事务状态1.4.联系1.2、特点1.3、原理2. Sprin

Redis实现分布式锁全解析之从原理到实践过程

《Redis实现分布式锁全解析之从原理到实践过程》:本文主要介绍Redis实现分布式锁全解析之从原理到实践过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录一、背景介绍二、解决方案(一)使用 SETNX 命令(二)设置锁的过期时间(三)解决锁的误删问题(四)Re

Gradle下如何搭建SpringCloud分布式环境

《Gradle下如何搭建SpringCloud分布式环境》:本文主要介绍Gradle下如何搭建SpringCloud分布式环境问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地... 目录Gradle下搭建SpringCloud分布式环境1.idea配置好gradle2.创建一个空的gr