分布式事务问题
1.1.本地事务

1.2.分布式事务

一旦有一个失败了,其他两个不知情失败的情况,还是执行并成功
在分布式系统下,一个业务跨越多个服务或数据源,每个服务都是一个分支事务,要保证所有分支事务最终状态一致,这样的事务就是分布式事务。
学习目标

2.1.CAP定理

 
 
2.2.BASE理论
 
2.3.解决分布式事务的思路
 
 
3.初识Seata
Seata是 2019 年 1 月份蚂蚁金服和阿里巴巴共同开源的分布式事务解决方案。致力于提供高性能和简单易用的分布式事务服务,为用户打造一站式的分布式解决方案。
官网地址:http://seata.io/,其中的文档、播客中提供了大量的使用说明、源码分析。
3.1.Seata的架构

 
3.2.部署TC服务
1.首先我们要下载seata-server包,地址在http://seata.io/zh-cn/blog/download.html
2.
 
3.

 
4.在nacos添加配置
 
配置内容:
# 数据存储方式,db代表数据库
 store.mode=db
 store.db.datasource=druid
 store.db.dbType=mysql
 store.db.driverClassName=com.mysql.jdbc.Driver
 store.db.url=jdbc:mysql://127.0.0.1:3306/seata?useUnicode=true&rewriteBatchedStatements=true
 store.db.user=root
 store.db.password=123
 store.db.minConn=5
 store.db.maxConn=30
 store.db.globalTable=global_table
 store.db.branchTable=branch_table
 store.db.queryLimit=100
 store.db.lockTable=lock_table
 store.db.maxWait=5000
 # 事务、日志等配置
 server.recovery.committingRetryPeriod=1000
 server.recovery.asynCommittingRetryPeriod=1000
 server.recovery.rollbackingRetryPeriod=1000
 server.recovery.timeoutRetryPeriod=1000
 server.maxCommitRetryTimeout=-1
 server.maxRollbackRetryTimeout=-1
 server.rollbackRetryTimeoutUnlockEnable=false
 server.undo.logSaveDays=7
 server.undo.logDeletePeriod=86400000
# 客户端与服务端传输方式
 transport.serialization=seata
 transport.compressor=none
 # 关闭metrics功能,提高性能
 metrics.enabled=false
 metrics.registryType=compact
 metrics.exporterList=prometheus
 metrics.exporterPrometheusPort=9898
其中的数据库地址、用户名、密码都需要修改成你自己的数据库信息
如果是新版的seata,我们修改yml文件:
config:
     # support: nacos, consul, apollo, zk, etcd3
     type: nacos
     nacos:
       server-addr: 127.0.0.1:8848
       namespace:
       group: SEATA_GROUP
       username: nacos
       password: nacos
       data-id: seataServer.properties
   registry:
     # support: nacos, eureka, redis, zk, consul, etcd3, sofa
     type: nacos
     nacos:
       application: seata-tc-server
       server-addr: 127.0.0.1:8848
       group: DEFAULT_GROUP      
       cluster: default
       username: nacos
       password: nacos
5.创建数据库表

SET NAMES utf8mb4;
 SET FOREIGN_KEY_CHECKS = 0;
-- ----------------------------
 -- 分支事务表
 -- ----------------------------
 DROP TABLE IF EXISTS `branch_table`;
 CREATE TABLE `branch_table`  (
   `branch_id` bigint(20) NOT NULL,
   `xid` varchar(128) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL,
   `transaction_id` bigint(20) NULL DEFAULT NULL,
   `resource_group_id` varchar(32) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
   `resource_id` varchar(256) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
   `branch_type` varchar(8) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
   `status` tinyint(4) NULL DEFAULT NULL,
   `client_id` varchar(64) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
   `application_data` varchar(2000) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
   `gmt_create` datetime(6) NULL DEFAULT NULL,
   `gmt_modified` datetime(6) NULL DEFAULT NULL,
   PRIMARY KEY (`branch_id`) USING BTREE,
   INDEX `idx_xid`(`xid`) USING BTREE
 ) ENGINE = InnoDB CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Compact;
-- ----------------------------
 -- 全局事务表
 -- ----------------------------
 DROP TABLE IF EXISTS `global_table`;
 CREATE TABLE `global_table`  (
   `xid` varchar(128) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL,
   `transaction_id` bigint(20) NULL DEFAULT NULL,
   `status` tinyint(4) NOT NULL,
   `application_id` varchar(32) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
   `transaction_service_group` varchar(32) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
   `transaction_name` varchar(128) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
   `timeout` int(11) NULL DEFAULT NULL,
   `begin_time` bigint(20) NULL DEFAULT NULL,
   `application_data` varchar(2000) CHARACTER SET utf8 COLLATE utf8_general_ci NULL DEFAULT NULL,
   `gmt_create` datetime NULL DEFAULT NULL,
   `gmt_modified` datetime NULL DEFAULT NULL,
   PRIMARY KEY (`xid`) USING BTREE,
   INDEX `idx_gmt_modified_status`(`gmt_modified`, `status`) USING BTREE,
   INDEX `idx_transaction_id`(`transaction_id`) USING BTREE
 ) ENGINE = InnoDB CHARACTER SET = utf8 COLLATE = utf8_general_ci ROW_FORMAT = Compact;
SET FOREIGN_KEY_CHECKS = 1;
6.启动TC服务

二、微服务集成seata
1.引入依赖 参与的每个服务都要引入
<dependency>
     <groupId>com.alibaba.cloud</groupId>
     <artifactId>spring-cloud-starter-alibaba-seata</artifactId>
     <exclusions>
         <!--版本较低,1.3.0,因此排除-->
         <exclusion>
             <artifactId>seata-spring-boot-starter</artifactId>
             <groupId>io.seata</groupId>
         </exclusion>
     </exclusions>
 </dependency>
 <!--seata starter 采用1.4.2版本-->
 <dependency>
     <groupId>io.seata</groupId>
     <artifactId>seata-spring-boot-starter</artifactId>
     <version>${seata.version}</version>
 </dependency>
2.修改配置文件
seata:
   registry: # TC服务注册中心的配置,微服务根据这些信息去注册中心获取tc服务地址
     # 参考tc服务自己的registry.conf中的配置
     type: nacos
     nacos: # tc
       server-addr: 127.0.0.1:8848
       namespace: ""
       group: DEFAULT_GROUP
       application: seata-tc-server # tc服务在nacos中的服务名称
       cluster: SH
   tx-service-group: seata-demo # 事务组,根据这个获取tc服务的cluster名称
   service:
     vgroup-mapping: # 事务组与TC服务cluster的映射关系
       seata-demo: SH

结合起来,TC服务的信息就是:public@DEFAULT_GROUP@seata-tc-server@SH,这样就能确定TC服务集群了。然后就可以去Nacos拉取对应的实例信息了。
4.动手实践
4.1.XA模式
XA 规范 是 X/Open 组织定义的分布式事务处理(DTP,Distributed Transaction Processing)标准,XA 规范 描述了全局的TM与局部的RM之间的接口,几乎所有主流的数据库都对 XA 规范 提供了支持。
4.1.1.两阶段提交

4.1.2.Seata的XA模型
 
 
4.1.3.实现XA模式

发现无论怎样,三个微服务都能成功回滚。
4.2.AT模式
AT模式同样是分阶段提交的事务模型,不过缺弥补了XA模型中资源锁定周期过长的缺陷。
4.2.1.Seata的AT模型

 
4.2.3.AT与XA的区别

4.2.4.脏写问题
在多线程并发访问AT模式的分布式事务时,有可能出现脏写问题,如图:

当阶段一结束,另一个线程就可以访问数据库操作,但是阶段二恢复数据会恢复到100,事务2的-10就没了
解决思路就是引入了全局锁的概念。在释放DB锁之前,先拿到全局锁。避免同一时刻有另外一个事务来操作当前数据。

非seata管理的另一个全局事务:

4.2.6.实现AT模式
AT模式中的快照生成、回滚等动作都是由框架自动完成,没有任何代码侵入,因此实现非常简单。
只不过,AT模式需要一个表来记录全局锁、另一张表来记录数据快照undo_log。

4.3.TCC模式
TCC模式与AT模式非常相似,每阶段都是独立事务,不同的是TCC通过人工编码来实现数据恢复。需要实现三个方法:
-  
Try:资源的检测和预留;
 -  
Confirm:完成资源操作业务;要求 Try 成功 Confirm 一定要能成功。
 -  
Cancel:预留资源释放,可以理解为try的反向操作。
 
4.3.1tcc模式原理

4.3.2.Seata的TCC模型

4.3.3.优缺点

4.3.4.事务悬挂和空回滚


业务分析

2)声明TCC接口
TCC的Try、Confirm、Cancel方法都需要在接口中基于注解来声明,
我们在account-service项目中的`cn.itcast.account.service`包中新建一个接口,声明TCC三个接口:

@LocalTCC
 public interface AccountTCCService {
    @TwoPhaseBusinessAction(name = "deduct", commitMethod = "confirm", rollbackMethod = "cancel")
     void deduct(@BusinessActionContextParameter(paramName = "userId") String userId,
                 @BusinessActionContextParameter(paramName = "money")int money);
boolean confirm(BusinessActionContext ctx);
    boolean cancel(BusinessActionContext ctx);
 }
3)编写实现类
@Service
@Slf4j
public class AccountTCCServiceImpl implements AccountTCCService {
    @Autowired
    private AccountMapper accountMapper;
    @Autowired
    private AccountFreezeMapper freezeMapper;
    @Override
    @Transactional
    public void deduct(String userId, int money) {
        // 0.获取事务id
        String xid = RootContext.getXID();
        //1.判断freeze中是否有冻结记录,如果有那么cancel执行过,要拒绝业务
        AccountFreeze oldFreeze = freezeMapper.selectById(xid);
        if (oldFreeze!=null){
            //cancel执行过了,拒绝业务,直接返回
            return;
        }
        // 1.扣减可用余额
        accountMapper.deduct(userId, money);
        // 2.记录冻结金额,事务状态
        AccountFreeze freeze = new AccountFreeze();
        freeze.setUserId(userId);
        freeze.setFreezeMoney(money);
        freeze.setState(AccountFreeze.State.TRY);
        freeze.setXid(xid);
        freezeMapper.insert(freeze);
    }
    @Override
    public boolean confirm(BusinessActionContext ctx) {
        // 1.获取事务id
        String xid = ctx.getXid();
        // 2.根据id删除冻结记录
        int count = freezeMapper.deleteById(xid);
        return count == 1;
    }
    @Override
    public boolean cancel(BusinessActionContext ctx) {
        // 0.查询冻结记录
        String xid = ctx.getXid();
        String userId = Objects.requireNonNull(ctx.getActionContext("userId")).toString();
        AccountFreeze freeze = freezeMapper.selectById(xid);
        //1.空回滚判断,判断freeze是否为null,为null证明try没执行需要空回滚
        if (freeze==null){
            //需要空回滚
            freeze=new AccountFreeze();
            freeze.setUserId(userId);
            freeze.setFreezeMoney(0);
            freeze.setState(AccountFreeze.State.CANCEL);
            freeze.setXid(xid);
            freezeMapper.insert(freeze);
            return true;
        }
        //2.幂等判断
        if (freeze.getState()==AccountFreeze.State.CANCEL){
            //已经处理过了,直接return
            return true;
        }
        // 3.恢复可用余额
        accountMapper.refund(freeze.getUserId(), freeze.getFreezeMoney());
        // 4.将冻结金额清零,状态改为CANCEL
        freeze.setFreezeMoney(0);
        freeze.setState(AccountFreeze.State.CANCEL);
        int count = freezeMapper.updateById(freeze);
        return count == 1;
    }
} 
4.4.SAGA模式
Saga 模式是 Seata 即将开源的长事务解决方案,将由蚂蚁金服主要贡献。
其理论基础是Hector & Kenneth 在1987年发表的论文Sagas。
Seata官网对于Saga的指南:Seata Saga 模式
4.4.1.原理

4.4.2.优缺点
优点:
-  
事务参与者可以基于事件驱动实现异步调用,吞吐高
 -  
一阶段直接提交事务,无锁,性能好
 -  
不用编写TCC中的三个阶段,实现简单
 
缺点:
-  
软状态持续时间不确定,时效性差
 -  
没有锁,没有事务隔离,会有脏写
 
4.5.四种模式对比
我们从以下几个方面来对比四种实现:
-  
一致性:能否保证事务的一致性?强一致还是最终一致?
 -  
隔离性:事务之间的隔离性如何?
 -  
代码侵入:是否需要对业务代码改造?
 -  
性能:有无性能损耗?
 -  
场景:常见的业务场景
 

一般默认都用AT 追求高性能就配合TCC
5.高可用
Seata的TC服务作为分布式事务核心,一定要保证集群的高可用性。
5.1.高可用架构模型

1.模拟异地容灾的TC集群

 
进入seata2/bin目录,然后运行命令: seata-server.bat -p 8092

 
2.将事务组映射配置到nacos

# 事务组映射关系
 service.vgroupMapping.seata-demo=SH
service.enableDegrade=false
 service.disableGlobalTransaction=false
 # 与TC服务的通信配置
 transport.type=TCP
 transport.server=NIO
 transport.heartbeat=true
 transport.enableClientBatchSendRequest=false
 transport.threadFactory.bossThreadPrefix=NettyBoss
 transport.threadFactory.workerThreadPrefix=NettyServerNIOWorker
 transport.threadFactory.serverExecutorThreadPrefix=NettyServerBizHandler
 transport.threadFactory.shareBossWorker=false
 transport.threadFactory.clientSelectorThreadPrefix=NettyClientSelector
 transport.threadFactory.clientSelectorThreadSize=1
 transport.threadFactory.clientWorkerThreadPrefix=NettyClientWorkerThread
 transport.threadFactory.bossThreadSize=1
 transport.threadFactory.workerThreadSize=default
 transport.shutdown.wait=3
 # RM配置
 client.rm.asyncCommitBufferLimit=10000
 client.rm.lock.retryInterval=10
 client.rm.lock.retryTimes=30
 client.rm.lock.retryPolicyBranchRollbackOnConflict=true
 client.rm.reportRetryCount=5
 client.rm.tableMetaCheckEnable=false
 client.rm.tableMetaCheckerInterval=60000
 client.rm.sqlParserType=druid
 client.rm.reportSuccessEnable=false
 client.rm.sagaBranchRegisterEnable=false
 # TM配置
 client.tm.commitRetryCount=5
 client.tm.rollbackRetryCount=5
 client.tm.defaultGlobalTransactionTimeout=60000
 client.tm.degradeCheck=false
 client.tm.degradeCheckAllowTimes=10
 client.tm.degradeCheckPeriod=2000
# undo日志配置
 client.undo.dataValidation=true
 client.undo.logSerialization=jackson
 client.undo.onlyCareUpdateColumns=true
 client.undo.logTable=undo_log
 client.undo.compress.enable=true
 client.undo.compress.type=zip
 client.undo.compress.threshold=64k
 client.log.exceptionRate=100
3.微服务读取nacos配置




















