分布式事务讲解 -消息队列+定时任务+本地事件表

1:使用场景介绍

这种方案是基于分布式服务理论从业务层面做出的设计,【消息队列+定时任务+本地事件表】方案就是消息队列搭配上本地事件表实现的分布式事务,这种方案的特点就是通过服务和消息队列的整合,可以对服务实现解耦,缺点是异步执行
比较适用于哪些业务简单,数据量小,并且不要求同步执行的场景。说白了,如果项目选型时,大部分人不会分布式事务框架,可以用这样的业务设计来提到

2:方案原理介绍

还是按照在电商系统中,调用支付系统完成后,然后支付系统调用订单系统更改订单状态

在这里插入图片描述
注:

在2-1,-2,-3这样的场景中,一般是先调用本地db,然后再调用第三方。
其实其他业务也一样,比如我们假设一个场景,如果我们需要调用第三方接口,还要执行我们自己的sql逻辑,那么开发时的顺序肯定是先执行调用第三方的,然后再执行我们的。毕竟我们自己的db可以通过事务回滚,第三方已经调走了,收不回来了;

3:开发步骤-生产者

1:pom

<?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 https://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>
    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>2.3.2.RELEASE</version>
        <relativePath/> <!-- lookup parent from repository -->
    </parent>
    <groupId>com.mashibing.serviceorder</groupId>
    <artifactId>service-order</artifactId>
    <version>0.0.1-SNAPSHOT</version>
    <name>service-order</name>
    <description>Demo project for Spring Boot</description>

    <properties>
        <java.version>1.8</java.version>
    </properties>

    <dependencies>

        <dependency>
            <groupId>org.projectlombok</groupId>
            <artifactId>lombok</artifactId>
            <optional>true</optional>
        </dependency>

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

        <!--activemq pool-->
        <dependency>
            <groupId>org.apache.activemq</groupId>
            <artifactId>activemq-pool</artifactId>
        </dependency>

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

        <dependency>
            <groupId>org.springframework.boot</groupId>
            <artifactId>spring-boot-starter-test</artifactId>
            <scope>test</scope>
            <exclusions>
                <exclusion>
                    <groupId>org.junit.vintage</groupId>
                    <artifactId>junit-vintage-engine</artifactId>
                </exclusion>
            </exclusions>
        </dependency>

        <!-- mysql:MyBatis相关依赖 -->
        <dependency>
            <groupId>org.mybatis.spring.boot</groupId>
            <artifactId>mybatis-spring-boot-starter</artifactId>
            <version>2.0.0</version>
        </dependency>

        <!-- mysql:mysql驱动 -->
        <dependency>
            <groupId>mysql</groupId>
            <artifactId>mysql-connector-java</artifactId>
        </dependency>

        <!-- mysql:阿里巴巴数据库连接池 -->
        <dependency>
            <groupId>com.alibaba</groupId>
            <artifactId>druid</artifactId>
            <version>1.1.12</version>
        </dependency>

        <!-- json-lib -->
        <dependency>
            <groupId>net.sf.json-lib</groupId>
            <artifactId>json-lib</artifactId>
            <version>2.4</version>
            <classifier>jdk15</classifier>
        </dependency>
    </dependencies>

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>

</project>

2:application配置


server:
  port: 5001

#应用名称及验证账号
spring:
  application:
    name: service-order
  activemq:
    broker-url: tcp://127.0.0.1:61616
    user: admin
    password: admin
    pool:
      enabled: true
      max-connections: 100

  datasource:
    type: com.alibaba.druid.pool.DruidDataSource
    driver-class-name: com.mysql.cj.jdbc.Driver
    url: jdbc:mysql://localhost:3306/service-order?characterEncoding=UTF-8&serverTimezone=Asia/Shanghai
    username: root
    password: root
    dbcp2:
      initial-size: 5
      min-idle: 5
      max-total: 5
      max-wait-millis: 200
      validation-query: SELECT 1
      test-while-idle: true
      test-on-borrow: false
      test-on-return: false

mybatis:
  mapper-locations:
  - classpath:mapper/*.xml

3:配置mq

// 配置消息发送端
@Configuration
public class ActiveMQConfig {
    @Value("${mqurl}")
    private String brokerUrl;
    // 创建队列,用于使用JMS发送消息
    @Bean
    public Queue queue() {
        return new ActiveMQQueue("ActiveMQQueue");
    }
    // 作为消息的发起方,创建一个消息队列连接工厂,供消费者连接和监听
    @Bean
    public ActiveMQConnectionFactory connectionFactory(){
        return new ActiveMQConnectionFactory(brokerUrl);
    }
}

4:配置定时任务,完成支付状态发入mq

@Component
public class ProduceTask {
    @Autowired
    private TblOrderEventDao tblOrderEventDao;
    @Autowired
    private Queue queue;
    @Autowired
    JmsMessagingTemplate jmsMessagingTemplate;
	// 创建一分钟一次的定时任务
    @Scheduled(cron="0 */1 * * * ?")
    @Transactional(rollbackFor = Exception.class)
    public void task(){
    	// 取到数据库本地事务表状态为【新建】的数据
        List<TblOrderEvent> tblOrderEventList = tblOrderEventDao.selectByOrderType("0");
        for (int i = 0; i < tblOrderEventList.size(); i++) {
            TblOrderEvent event = tblOrderEventList.get(i);
            // 更改这条数据的状态为【已发送】
            tblOrderEventDao.updateEvent(event.getOrderType());
            System.out.println("修改数据库完成");
			// 使用jmsMessagingTemplate发送信息至消息队列
            jmsMessagingTemplate.convertAndSend(	
            	queue,
            	JSONObject.fromObject(event).toString()
            );
        }
    }
}

5:开启启动类注解

@SpringBootApplication
@EnableJms // 支持消息传递的注解
@EnableScheduling   // 支持定时任务的注解
public class ServiceOrderApplication {
    public static void main(String[] args) {
        SpringApplication.run(ServiceOrderApplication.class, args);
    }
}

4:开发步骤-消费者

1-2 基本配置一样

3:配置mq,包括重试机制

// 配置消息消费端
@Configuration
public class ActiveMQConfig {
    @Value("${mqurl}")
    private String brokerUrl;
    @Value("${mquser}")
    private String user;
    @Value("${mqpassword}")
    private String password;

    /**
     * 作为消费方,根据activeMQ地址连接MQ工厂并设置重发机制
     * @param redeliveryPolicy
     * @return
     */
    @Bean
    public ActiveMQConnectionFactory connectionFactory
    (RedeliveryPolicy redeliveryPolicy){
        ActiveMQConnectionFactory activeMQConnectionFactory = 
        new ActiveMQConnectionFactory(user,password,brokerUrl);
        activeMQConnectionFactory.setRedeliveryPolicy(redeliveryPolicy);
        return activeMQConnectionFactory;
    }
    /**
     * 如果一个消息消费失败了,MQ工厂重发消息供消费者消费
     * @return
     */
    @Bean
    public RedeliveryPolicy redeliveryPolicy(){
        RedeliveryPolicy redeliveryPolicy = new RedeliveryPolicy();
        return redeliveryPolicy;
    }
    /**
     * 设置消息队列 确认机制
     * 设置手动确认,消费者确认之后才表明消费成功
     * @param activeMQConnectionFactory
     * @return
     */
    @Bean
    public JmsListenerContainerFactory jmsListenerContainerFactory
    (ActiveMQConnectionFactory activeMQConnectionFactory){
        DefaultJmsListenerContainerFactory bean = 
        new DefaultJmsListenerContainerFactory();
        bean.setConnectionFactory(activeMQConnectionFactory);
        // 1: 自动确认,2: 客户端手动确认,3:自动批量确认,4 事务提交并确认。
        bean.setSessionAcknowledgeMode(2);
        return bean;
    }
}

4:配置消费者消费

@Component
public class ConsumerQueue {
    @Autowired
    private TblPayEventDao tblPayEventDao;
    // destination属性值意为监听的目标,来源于消息发起者创建消息队列的nama属性值。
    // containerFactory属性值意为容器工厂,来源于消息消费者配置的监听消息的容器工厂。
    @JmsListener(destination = "ActiveMQQueue",
    	containerFactory = "jmsListenerContainerFactory")
    public void receive(TextMessage textMessage, Session session) 
    	throws JMSException {
        try {
            System.out.println("收到的消息:"+textMessage.getText());
            String content = textMessage.getText();
			// 将传来信息转成对象并操作数据库
            TblPayEvent tblPayEvent = 
            	(TblPayEvent) JSONObject.toBean(
            		JSONObject.fromObject(content), TblPayEvent.class);
            tblPayEventDao.insert(tblPayEvent);
            // 业务完成,确认消息 消费成功
            // 因为设置了手动确认,所以,需要调用消息的acknowledge方法
            textMessage.acknowledge();
        }catch (Exception e){
            // 回滚消息
            e.printStackTrace();
            System.out.println("异常了");
            // 消息消费失败,恢复回消息队列,等待重发再次消费。
            session.recover();
        }

    }
    /**
     * 监听死信队列,按需写方法内的业务逻辑
     * 补偿 处理(人工,脚本)。自己根据自己情况。
     * @param text
     */
    @JmsListener(destination = "DLQ.ActiveMQQueue")
    public void receive2(String text){
        //发邮件,人工干预
    }
}

  • 0
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
### 回答1: STM32是一款非常强大的微控制器,它可以实现许多复杂的应用。其中,搭载RTOS系统可以实现任务、消息队列、串口通信等功能。 RTOS是一种实时操作系统,它可以实现多任务处理、任务优先级控制、资源共享等功能。在STM32上搭载RTOS系统之后,可以很容易地实现多任务处理。不同的任务可以并行地执行,从而提高系统的响应速度。 消息队列是一种重要的通信机制。在STM32上,可以使用RTOS系统中的消息队列功能,实现任务间的数据传输。通过消息队列机制,可以实现任务间的协作,提高系统的整体效率。 串口通信是一种常见的通信方式。在STM32上,可以通过使用HAL库中的串口功能,实现串口通信。使用RTOS系统可以更好地管理串口通信任务,避免因串口数据传输速度慢而导致的死锁等问题。 综上所述,STM32搭载RTOS系统可以实现任务、消息队列、串口通信等功能。这些功能可以优化系统性能,提高系统响应速度,是嵌入式系统开发的重要工具。 ### 回答2: STM32是一种现代的、高性能的微处理器,它可以在各类嵌入式系统中使用。在开发嵌入式系统时,使用操作系统(即RTOS)能极大地提高开发效率和系统及时性、可靠性和可维护性。本文将介绍如何使用STM32搭载RTOS来实现任务、消息队列和串口通信。 一、搭载RTOS RTOS(Real-time Operating System)是嵌入式系统的一种操作系统,其目标在于将各类程序任务和实时数据处理转移到系统级别上。STM32可以使用多种不同的RTOS,如FreeRTOS、uC/OS-II、ThreadX等等。这里我们选用FreeRTOS来展示如何实现任务、消息队列和串口通信。 二、实现任务 任务通常是RTOS中的基础单元,通过任务管理器来调度不同的任务。在使用FreeRTOS创建任务时,需要使用vTaskCreate函数来创建任务并指定任务运行的优先级、堆栈大小和任务处理函数。如下是一个简单的任务处理函数示例: void vMyTask(void *pvParameters) { for(;;) { //任务处理代码 } } 任务处理函数会不断循环执行任务处理代码,直到任务被删除或终止。 三、实现消息队列 消息队列是管理任务间通信的重要机制之一。它允许任务之间传输数据和状态信息。在FreeRTOS中,消息队列可以使用xQueueCreate函数创建并指定队列容量。下面是一个简单的消息队列创建示例: xQueueHandle xQueue; xQueue = xQueueCreate(5, sizeof(int)); 上述代码创建了一个容量为5的队列,并在创建时指定了队列传输的数据类型为int型。 四、实现串口通信 串口通信是一种基本的通信方法,通常用于嵌入式系统中设备的调试和监测。在STM32中,实现串口通信可以使用STM32 HAL库中的UART模块。使用HAL库可以简化串口模块的原始寄存器操作。下面是一个简单的UART初始化函数示例: void UART_Init(void) { huart1.Instance = USART1; huart1.Init.BaudRate = 115200; huart1.Init.WordLength = UART_WORDLENGTH_8B; huart1.Init.StopBits = UART_STOPBITS_1; huart1.Init.Parity = UART_PARITY_NONE; huart1.Init.Mode = UART_MODE_TX_RX; huart1.Init.HwFlowCtl = UART_HWCONTROL_NONE; huart1.Init.OverSampling = UART_OVERSAMPLING_16; HAL_UART_Init(&huart1); } 本示例采用了USART1串口并将波特率设置为115200,其中其他参数的设置可以根据具体的应用场景进行调整。 以上就是使用STM32搭载RTOS实现任务、消息队列和串口通信的基本内容。通过使用RTOS,可以极大地提高系统的实时性、可靠性和可维护性。但需要注意的是,在使用RTOS的过程中,需要遵循一些基本的开发原则,以避免各种问题和错误。 ### 回答3: STM32是一款嵌入式控制器,可以实现任务、消息队列和串口通信等功能。其中最为常用的方法是使用操作系统(RTOS)来实现这些功能。RTOS是一种可重入可中断的操作系统,它提供了一些可靠的服务,如任务管理、时间管理、信号量、消息队列等,它可以大大简化嵌入式软件的设计过程,并增加了系统的可靠性和可维护性。 实现任务管理的方法是使用RTOS的任务管理功能来创建、调度和协调任务。任务是通过分配固定的时间片被调度来执行。在STM32中,任务调度器是一个内核线程,它管理所有任务。任务可以是相互独立、相互协作或同步协作的。任务可以优先级别,高优先级任务可以抢占低优先级任务的执行时间,这样可以保证任务的实时性。 实现消息队列的方法是使用RTOS的消息队列功能。消息队列是一种数据结构,用于存储消息。它具有FIFO(先入先出)队列的特性,也具有可靠性和高度可扩展性。在STM32中,可以使用消息队列来异步传递数据和消息,这样可以降低任务间的耦合度,提高任务调度器的可靠性。 实现串口通信的方法是通过使用RTOS的中断和任务管理系统来实现。在STM32中,串口是通过一个内核线程来处理的,这个线程可以接收和发送数据。此线程可以通过使用任务和时间管理,以及中断和DMA来实现。中断处理程序可以异步地处理串口游泳和接收事件,并将其推送到消息队列中。 总之,在STM32中,使用RTOS来实现任务、消息队列和串口通信,可以简化嵌入式设计的过程,并提高系统的可靠性和可维护性。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

苍煜

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

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

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

打赏作者

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

抵扣说明:

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

余额充值