分布式事务 LCN

第一节 分布式事务的产生

  • 数据库分库分表就产生了分布式事务;

  • 项目拆分服务化也产生了分布式事务;

当单个数据库的性能产生瓶颈的时候,可能会对数据库进行分区(物理分区),分区之后可能不同的库就处于不同的服务器上了,此时单个数据库的ACID已经不能适应这种情况了,在集群环境下,再想保证ACID几乎是很难达到,这时就需要引入一个新的理论来适应这种集群的情况,这个理论就是CAP定理

第二节 CAP定理

CAP定理是由加州大学伯克利分校Eric Brewer教授提出来的,他指出WEB服务无法同时满足以下3个属性:

  • 一致性(Consistency) : 客户端知道一系列的操作都会同时发生

  • 可用性(Availability) : 每个操作都必须以可预期的响应结束

  • 分区容错性(Partition tolerance) : 即使出现单个组件无法可用,操作依然可以完成

在分布式系统中的任何数据库设计,一个Web应用至多只能同时支持上面的两个属性。分区容错性是分布式系统必须要保证的,因此,设计人员必须在一致性与可用性之间做出选择。对于任何系统来说,可用性都是比较重要的,因此,在选择的时候,基本上都会选择可用性。那么,一致性就得不到保证,为了解决这个问题,也需要引入一个新的 理论来适应这种情况,这个理论就是BASE理论。

第三节 BASE 理论

BASE理论指的是:

  • Basically Available(基本可用)

  • Soft state(软状态)

  • Eventually consistency(最终一致性)

BASE理论是对CAP中的一致性和可用性进行一个权衡的结果,理论的核心思想就是:既然无法做到强一致,但每个应用都可以根据自身的业务特点,采用适当的方式来使系统达到最终一致性

第四节 XA 协议(XA Transactions)

XA 是一个两阶段提交协议,该协议分为以下两个阶段:

  • 第一阶段:事务协调器要求每个涉及到事务的数据库预提交( precommit )此操作,并反映是否可以提交.

  • 第二阶段:事务协调器要求每个数据库提交数据。

其中,如果有任何一个数据库否决此次提交,那么所有数据库都会被要求回滚它们在此事务中的那部分信息。

第五节 分布式事务解决方案

1. 两阶段提交协议

1.1 概念

两阶段提交协议,简写2PC2PC就是使用 XA 协议的原理。通过引入协调者(Coordinator)来协调参与者的行为,并最终决定这些参与者是否要真正执行事务。

1.2 执行流程
  • 准备阶段:协调者询问参与者事务是否执行成功,参与者发回事务执行结果。

  • 提交阶段:如果事务在每个参与者上都执行成功,事务协调者发送通知让参与者提交事务;否则,协调者发送通知让参与者回滚事务。需要注意的是,在准备阶段,参与者执行了事务,但是还未提交。只有在提交阶段接收到协调者发来的通知后,才进行提交或者回滚。

1.3 存在问题
  • 所有事务参与者在等待其它参与者响应的时候都处于同步阻塞状态,无法进行其它操作

  • 协调者在 2PC 中起到非常大的作用,发生故障将会造成很大影响。特别是在阶段二发生故障,所有参与者会一直等待状态,无法完成其它操作。

  • 在阶段二,如果协调者只发送了部分 Commit 消息,此时网络发生异常,那么只有部分参与者接收到 Commit 消息,也就是说只有部分参与者提交了事务,使得系统数据不一致。

  • 太过保守 任意一个节点失败就会导致整个事务失败,没有完善的容错机制。

2. 补偿事务(TCC

2.1 概念

TCC 其实就是采用的补偿机制,其核心思想是:针对每个操作,都要注册一个与其对应的确认和补偿(撤销)操作。TCC 一共包括两个阶段 Try阶段、 Confirm/Cancel阶段

2.2 执行流程
  • Try 阶段主要是对业务系统做检测及资源预留

  • Confirm 阶段主要是对业务系统做确认提交,Try阶段执行成功并开始执行 Confirm阶段时,默认 Confirm阶段是不会出错的。只要Try成功,Confirm一定成功。

  • Cancel 阶段主要是在业务执行错误,需要回滚的状态下执行的业务取消,预留资源释放。

2.3 问题
  • 代码侵入多,实现复杂

  • 容易出现空回滚,需要使用代码解决

  • 容易出现事务重试,需要考虑幂等性,也需要使用代码解决

  • 容易出现资源悬挂,同样需要使用代码解决

第六节 LCN分布式事务框架

1. 什么是LCN分布式事务

LCN分布式事务框架其本身并不创建事务,而是基于对本地事务的协调从而达到事务一致性的效果。LCN常用的分布式事务有两种模式:LCN模式TCC模式

1.1 LCN模式

LCN 模式是通过代理 Connection 的方式实现对本地事务的操作,然后在由 TxManager 统一协调控制事务。当本地事务提交回滚或者关闭连接时将会执行假操作,该代理的连接将由 LCN 连接池管理。

LCN模式的特点:

  • 对代码的嵌入性为低。

  • 仅限于本地存在连接对象且可通过连接对象控制事务的模块。

  • 事务提交与回滚是由本地事务方控制,对于数据一致性上有较高的保障。

  • 代理的连接需要随事务发起方一共释放连接,增加了连接占用的时间。

1.2 TCC模式

TCC事务机制相对于传统事务机制(X/Open XA Two-Phase-Commit),其特征在于它不依赖资源管理器(RM)对 XA 的支持,而是通过对(由业务系统提供的)业务逻辑的调度来实现分布式事务。主要由三步操作,Try : 尝试执行业务、 Confirm : 确认执行业务、Cancel : 取消执行业务。

TCC模式的特点:

  • 对代码的嵌入性高,要求每个业务需要写三种步骤的操作。

  • 对有无本地事务控制都可以支持使用面广。

  • 数据一致性控制几乎完全由开发者控制,对业务开发难度要求高。

2. LCN分布式事务原理

LCN事务控制原理是由事务模块TxClient下的代理连接池与 TxManager 的协调配合完成的事务协调控制。TxClient 的代理连接池实现了 javax.sql.DataSource 接口,并重写了 close 方法,事务模块在提交关闭以后 TxClient 连接池将执行"假关闭"操作,等待 TxManager 协调完成事务以后在关闭连接。

核心步骤

  • 创建事务组 是指在事务发起方开始执行业务代码之前先调用 TxManager 创建事务组对象,然后拿到事务标示 GroupId的过程。

  • 添加事务组 添加事务组是指参与方在执行完业务方法以后,将该模块的事务信息添加通知给 TxManager 的操作。

  • 关闭事务组 是指在发起方执行完业务代码以后,将发起方执行结果状态通知给 TxManager 的动作。当执行完关闭事务组的方法以后,TxManager 将根据事务组信息来通知相应的参与模块提交或回滚事务。

第三节 LCN分布式事务实现

1. LCN 服务器搭建

1.1 编译源码
mvn install -Dmaven.test.skip=true
1.2 创建父工程 qf-cloud-tx
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>com.qf</groupId>
    <artifactId>spring-cloud-tx</artifactId>
    <version>1.0</version>
    <dependencyManagement>
        <dependencies>
            <dependency>
                <groupId>org.springframework.cloud</groupId>
                <artifactId>spring-cloud-dependencies</artifactId>
                <version>Finchley.SR3</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
            <dependency>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-dependencies</artifactId>
                <version>2.0.1.RELEASE</version>
                <type>pom</type>
                <scope>import</scope>
            </dependency>
        </dependencies>
    </dependencyManagement>
</project>
1.3 创建子工程 qf-lcn
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <artifactId>spring-cloud-tx</artifactId>
        <groupId>com.qf</groupId>
        <version>1.0</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>qf-lcn</artifactId>

    <dependencies>
        <!-- 参照例子引入需要的依赖jar -->
        <dependency>
            <groupId>com.codingapi.txlcn</groupId>
            <artifactId>txlcn-tm</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>
    </dependencies>
</project>
1.4 配置 application.properties
spring.application.name=txlcn-tm
server.port=7970

spring.datasource.driver-class-name=com.mysql.jdbc.Driver
spring.datasource.url=jdbc:mysql://127.0.0.1:3306/tx-manager?characterEncoding=UTF-8
spring.datasource.username=root
spring.datasource.password=root
spring.jpa.database-platform=org.hibernate.dialect.MySQL5InnoDBDialect
spring.jpa.hibernate.ddl-auto=update


# TM后台登陆密码
tx-lcn.manager.admin-key=123456

tx-lcn.manager.host=127.0.0.1
tx-lcn.manager.port=8070

# 开启日志,默认为false
tx-lcn.logger.enabled=true
tx-lcn.logger.driver-class-name=${spring.datasource.driver-class-name}
tx-lcn.logger.jdbc-url=${spring.datasource.url}
tx-lcn.logger.username=${spring.datasource.username}
tx-lcn.logger.password=${spring.datasource.password}
logging.level.com.codingapi.txlcn=DEBUG

#redis 主机
spring.redis.host=127.0.0.1
#redis 端口
spring.redis.port=6379
#redis 密码
spring.redis.password=
1.5 编写启动类
package com.qf.lcn;

import com.codingapi.txlcn.tm.config.EnableTransactionManagerServer;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;

@SpringBootApplication
@EnableTransactionManagerServer
public class QfLcnServer {

    public static void main(String[] args) {
        SpringApplication.run(QfLcnServer.class, args);
    }
}

2. 搭建注册中心

2.1 创建子工程 qf-registry
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <artifactId>spring-cloud-tx</artifactId>
        <groupId>com.qf</groupId>
        <version>1.0</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>qf-registry</artifactId>

    <dependencies>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-server</artifactId>
        </dependency>
    </dependencies>
</project>
2.2 配置 application.yml
server:
  port: 11000
eureka:
  client:
    fetch-registry: false
    register-with-eureka: false
    service-url:
      defaultZone: http://localhost:11000/eureka
2.3 编写启动类
package com.qf.registry;

import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.server.EnableEurekaServer;

@SpringBootApplication
@EnableEurekaServer
public class QfRegistry {

    public static void main(String[] args) {
        SpringApplication.run(QfRegistry.class, args);
    }
}

3. 搭建商品服务

3.1创建子工程 qf-goods
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <artifactId>spring-cloud-tx</artifactId>
        <groupId>com.qf</groupId>
        <version>1.0</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>qf-goods</artifactId>

    <dependencies>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>2.2.1</version>
        </dependency>

        <dependency>
            <groupId>com.codingapi.txlcn</groupId>
            <artifactId>txlcn-tc</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>com.codingapi.txlcn</groupId>
            <artifactId>txlcn-txmsg-netty</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.17</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-openfeign</artifactId>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid</artifactId>
            <version>1.2.8</version>
        </dependency>
    </dependencies>
</project>
3.2 配置 application.yml
server:
  port: 13000
eureka:
  client:
    service-url:
      defaultZone: http://localhost:11000/eureka/

spring:
  application:
    name: qf-goods
  datasource:
    driver-class-name: com.mysql.cj.jdbc.Driver
    url: jdbc:mysql://localhost:3306/goods?serverTimezone=UTC
    username: root
    password: root
    type: com.alibaba.druid.pool.DruidDataSource
mybatis:
  mapper-locations: classpath:mapper/*Mapper.xml
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
tx-lcn:
  primary-key-package:
  client:
    manager-address: 127.0.0.1:8070
  #是否启动LCN负载均衡策略(优化选项,开启与否,功能不受影响)
  ribbon:
    loadbalancer:
      dtx:
        enabled: true
  #是否开启日志记录,当开启以后需要配置对应logger的数据库连接配置信息
  logger:
    enabled: true
    driver-class-name: ${spring.datasource.driver-class-name}
    jdbc-url: ${spring.datasource.url}
    username: ${spring.datasource.username}
    password: ${spring.datasource.password}
3.3 编写启动类
package com.qf.goods;

import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
import org.springframework.cloud.openfeign.EnableFeignClients;

@SpringBootApplication
@EnableEurekaClient
@EnableFeignClients
@MapperScan("com.qf.goods.mapper")
public class QfGoodsServer {

    public static void main(String[] args) {
        SpringApplication.run(QfGoodsServer.class, args);
    }
}
3.4 编写 GoodsMapper 接口
package com.qf.goods.mapper;

import org.apache.ibatis.annotations.Mapper;
import org.apache.ibatis.annotations.Param;
import org.apache.ibatis.annotations.Update;

import java.util.Map;

@Mapper
public interface GoodsMapper {

    @Update("UPDATE goods SET num = num - #{goods.buyCount} WHERE id = #{goods.goodsId}")
    int buyGoods(@Param("goods") Map<String,Object> params);
}
3.5 编写业务层接口及实现
package com.qf.goods.service;

import java.util.Map;

public interface GoodsService {

    int updateGoods(Map<String,Object> params);
}


package com.qf.goods.service.impl;

import com.qf.goods.mapper.GoodsMapper;
import com.qf.goods.service.GoodsService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import java.util.Map;

@Service
public class GoodsServiceImpl implements GoodsService {

    @Autowired
    private GoodsMapper goodsMapper;

    @Transactional
    @Override
    public int buyGoods(Map<String, Object> params) {
        return goodsMapper.buyGoods(params);
    }
}
3.6 编写控制器
package com.qf.goods.controller;

import com.qf.goods.service.GoodsService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PutMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.Map;

@RestController
@RequestMapping("/goods")
public class GoodsController {


    @Autowired
    private GoodsService goodsService;

    @PutMapping
    public int buyGoods(@RequestBody Map<String,Object> params){
        return goodsService.buyGoods(params);
    }
}

4. 搭建订单服务

4.1 创建子工程 qf-order
<?xml version="1.0" encoding="UTF-8"?>
<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <parent>
        <artifactId>spring-cloud-tx</artifactId>
        <groupId>com.qf</groupId>
        <version>1.0</version>
    </parent>
    <modelVersion>4.0.0</modelVersion>

    <artifactId>qf-order</artifactId>

    <dependencies>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-netflix-eureka-client</artifactId>
        </dependency>

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-web</artifactId>
        </dependency>

        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>2.2.1</version>
        </dependency>

        <dependency>
            <groupId>com.codingapi.txlcn</groupId>
            <artifactId>txlcn-tc</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>
        <dependency>
            <groupId>com.codingapi.txlcn</groupId>
            <artifactId>txlcn-txmsg-netty</artifactId>
            <version>5.0.2.RELEASE</version>
        </dependency>

        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
            <version>8.0.17</version>
        </dependency>
        <dependency>
            <groupId>org.springframework.cloud</groupId>
            <artifactId>spring-cloud-starter-openfeign</artifactId>
        </dependency>
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid</artifactId>
            <version>1.2.8</version>
        </dependency>
    </dependencies>
</project>
4.2 配置 application.yml
server:
  port: 12000
eureka:
  client:
    service-url:
      defaultZone: http://localhost:11000/eureka/

spring:
  application:
    name: qf-order
  datasource:
    driver-class-name: com.mysql.cj.jdbc.Driver
    url: jdbc:mysql://localhost:3306/order?serverTimezone=UTC
    username: root
    password: root
    type: com.alibaba.druid.pool.DruidDataSource
mybatis:
  mapper-locations: classpath:mapper/*Mapper.xml
  configuration:
    log-impl: org.apache.ibatis.logging.stdout.StdOutImpl
tx-lcn:
  client:
    manager-address: 127.0.0.1:8070
  primary-key-package:
  #是否启动LCN负载均衡策略(优化选项,开启与否,功能不受影响)
  ribbon:
    loadbalancer:
      dtx:
        enabled: true
#是否开启日志记录,当开启以后需要配置对应logger的数据库连接配置信息
  logger:
    enabled: true
    driver-class-name: ${spring.datasource.driver-class-name}
    jdbc-url: ${spring.datasource.url}
    username: ${spring.datasource.username}
    password: ${spring.datasource.password}
4.3 编写启动类
package com.qf.order;

import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;

@SpringBootApplication
@EnableEurekaClient
@MapperScan("com.qf.order.mapper")
public class QfOrderServer {

    public static void main(String[] args) {
        SpringApplication.run(QfOrderServer.class, args);
    }
}
4.4 编写 OrderMapper 接口
package com.qf.order.mapper;

import org.apache.ibatis.annotations.*;

import java.util.Map;

@Mapper
public interface OrderMapper {

    @Insert("INSERT INTO orders(goods_id,buy_count,payment) VALUES(#{order.goodsId}, #{order.buyCount}, #{order.payment})")
    @SelectKey(keyProperty = "order.id", before = false, statement = "SELECT LAST_INSERT_ID()", resultType = Long.class)
    int addOrder(@Param("order") Map<String,Object> params);
}
4.5 编写业务层接口及实现
package com.qf.order.service;

import java.util.Map;

public interface OrderService {

    int addOrder(Map<String,Object> params);
}


package com.qf.order.service.impl;

import com.qf.order.mapper.OrderMapper;
import com.qf.order.service.OrderService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import java.util.Map;

@Service
public class OrderServiceImpl implements OrderService {

    @Autowired
    private OrderMapper orderMapper;

    @Transactional
    @Override
    public int addOrder(Map<String, Object> params) {
        return orderMapper.addOrder(params);
    }
}
4.6 编写控制层
package com.qf.order.controller;

import com.qf.order.service.OrderService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RestController;

import java.util.Map;

@RestController
@RequestMapping("/order")
public class OrderController {

    @Autowired
    private OrderService orderService;

    @PostMapping
    public int addOrder(@RequestBody Map<String,Object> params){
        return orderService.addOrder(params);
    }
}

现有有一用户购买了商品,购买之后需要产生订单,如何保证订单的正确创建?

5. LCN模式实现

5.1 在 qf-goods 工程中添加Feign接口
package com.qf.goods.client;

import org.springframework.cloud.openfeign.FeignClient;
import org.springframework.web.bind.annotation.PostMapping;
import org.springframework.web.bind.annotation.RequestBody;

import java.util.Map;

@FeignClient(name = "qf-order", path = "/order")
public interface OrderClient {

    @PostMapping
    int addOrder(@RequestBody Map<String,Object> params);
}
5.2 修改qf-goods 业务实现
package com.qf.goods.service.impl;

import com.codingapi.txlcn.tc.annotation.DTXPropagation;
import com.codingapi.txlcn.tc.annotation.LcnTransaction;
import com.qf.goods.client.OrderClient;
import com.qf.goods.mapper.GoodsMapper;
import com.qf.goods.service.GoodsService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import java.util.Map;

@Service
public class GoodsServiceImpl implements GoodsService {

    @Autowired
    private GoodsMapper goodsMapper;

    @Autowired
    private OrderClient orderClient;

    //开启LCN事务
    @LcnTransaction(propagation = DTXPropagation.REQUIRED)
    @Transactional
    @Override
    public int buyGoods(Map<String, Object> params) {
        int result1 = goodsMapper.buyGoods(params);
        double price = (double) params.remove("price");
        int buyCount = (int) params.get("buyCount");
        double payment = price * buyCount;
        params.put("payment", payment);
        int result2 = orderClient.addOrder(params);
        return result1 == 1 && result2 == 1 ? 1 : 0;
    }
}
5.3 修改qf-goods 启动类
package com.qf.goods;

import com.codingapi.txlcn.tc.config.EnableDistributedTransaction;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.client.circuitbreaker.EnableCircuitBreaker;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;
import org.springframework.cloud.openfeign.EnableFeignClients;

@SpringBootApplication
@EnableDistributedTransaction //启用分布式事务
@EnableEurekaClient
@EnableFeignClients
@EnableCircuitBreaker
@MapperScan("com.qf.erp.mapper")
public class QfGoodsServer {

    public static void main(String[] args) {
        SpringApplication.run(QfGoodsServer.class, args);
    }
}
5.4 修改 qf-order 业务层
package com.qf.order.service.impl;

import com.codingapi.txlcn.tc.annotation.DTXPropagation;
import com.codingapi.txlcn.tc.annotation.LcnTransaction;
import com.qf.order.mapper.OrderMapper;
import com.qf.order.service.OrderService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.stereotype.Service;
import org.springframework.transaction.annotation.Transactional;

import java.util.Map;

@Service
public class OrderServiceImpl implements OrderService {

    @Autowired
    private OrderMapper orderMapper;

    @LcnTransaction(propagation = DTXPropagation.SUPPORTS) //开启LCN事务
    @Transactional
    @Override
    public int addOrder(Map<String, Object> params) {
        return orderMapper.addOrder(params);
    }
}
5.5 修改qf-order 启动类
package com.qf.order;

import com.codingapi.txlcn.tc.config.EnableDistributedTransaction;
import org.mybatis.spring.annotation.MapperScan;
import org.springframework.boot.SpringApplication;
import org.springframework.boot.autoconfigure.SpringBootApplication;
import org.springframework.cloud.netflix.eureka.EnableEurekaClient;

@SpringBootApplication
@EnableEurekaClient
@EnableDistributedTransaction //开启分布式事务
@MapperScan("com.qf.order.mapper")
public class QfOrderServer {

    public static void main(String[] args) {
        SpringApplication.run(QfOrderServer.class, args);
    }
}

6. TCC模式

评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

十点 vha

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值