threadX 消息队列

2024-06-05 03:44
文章标签 队列 消息 threadx

本文主要是介绍threadX 消息队列,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1、 使用消息列的目的

在ThreadX操作系统下使用消息队列的目的主要有以下几点:

  1. 提高CPU利用率
    • 消息队列是RTOS(实时操作系统)中常用的一种数据通信方式,常用于任务与任务之间或是中断与任务之间的数据传递。
    • 相比裸机系统中使用全局变量进行数据传递需要不断地轮询标志状态,使用RTOS的消息队列具有任务阻塞机制。当没有需要处理的消息时任务挂起等待消息,此时其他任务可以占用CPU执行其他操作,当有消息放入队列时任务恢复运行进行消息接收和处理。这种消息处理机制相比裸机而言大大地提高了CPU利用率。
  2. 数据传递的灵活性
    • ThreadX的消息队列可以传递任意长度的数据,因为它是采用传递数据指针的方式,这使得在数据传递上具有更高的灵活性。
  3. 支持紧急消息处理
    • ThreadX的消息队列支持“消息置顶通知”功能,可以将消息放在队列的最前面,使得任务可以及时处理某些紧急消息。
  4. 解决线程间的通信问题
    • 消息队列是ThreadX中线程间(以及线程和中断间)通信的主要手段。使用消息队列可以有效管理线程,解决中断服务程序与线程之间消息传递的问题,并且避免了多线程访问冲突的问题。
  5. FIFO机制
    • 消息队列采用FIFO(先进先出)机制,更有利于数据的处理。
  6. 异步通信和解耦
    • 与其他消息队列系统类似,ThreadX下的消息队列也允许生产者发送消息后立即继续执行,而消费者可以在适当的时候处理消息,这实现了异步通信。同时,消息队列将生产者和消费者解耦,使它们可以独立扩展和维护。
  7. 流量削峰
    • 在高并发场景下,消息队列可以平滑处理流量峰值,避免系统因瞬时压力而崩溃。
  8. 可靠性
    • 消息队列通常提供了高度可靠的消息传递保证,确保消息可以被传递一次且仅一次。

综上所述,ThreadX操作系统下使用消息队列的目的是为了提高系统的性能、灵活性、可靠性和可扩展性。

2、Thread X 任务 间 消息队列的 实现

        

  • 注意消息队列的 send 和recv速度,否则可能会造成消息队列存满的情况
  • 可以在tx_queue_receive函数中设置超时等待

3、中断方式

        

  • 中断服务程序中调用发送函数,一定要设置超时形参为TX_NO_WAIT。
  • 中断函数的执行时间越短越好,防止其它低于这个中断优先级的异常不能得到及时响应。
  • 实际应用中,建议不要在中断中实现消息处理,用户可以在中断服务程序里面发送消息通知任务,在任务中实现消息处理,这样可以有效地保证中断服务程序的实时响应。同时此任务也需要设置为高优先级,以便退出中断函数后任务可以得到及时执行。

4、消息对列的主要函数

消息队列创建

UINT tx_queue_create(
TX_QUEUE *queue_ptr,  //消息队列控制块
CHAR *name_ptr,            //消息队列名字。
UINT message_size,       //消息队列每个消息的大小
VOID *queue_start,      //消息队列缓冲地址 必须保证此地址4字节对齐
ULONG queue_size);  //消息缓冲大小,单位字节

消息队列发送函数

UINT tx_queue_send(
TX_QUEUE *queue_ptr,  //消息队列控制块
VOID *source_ptr,         //要发送的数据地址。
ULONG wait_option);   //等待选项

消息队列接收函数

UINT tx_queue_receive(
TX_QUEUE *queue_ptr,   //消息队列控制块
VOID *destination_ptr,    //消息队列复制出来数据的存储地址
ULONG wait_option);     //等待选项

等待选项: 如果消息队列空了,这个形参将派上用场:

  • TX_NO_WAIT (0x00000000),表示不管消息队列是否空,立即返回。如果在定时器组,初始化或中断里面调用,必须要设置成这个参数。
  • TX_WAIT_FOREVER (0xFFFFFFFF),表示永久等待,直到消息队列有数据。
  • 等待时间,范围0x00000001 到 0xFFFFFFFE,单位系统时钟节拍、

5、使用threadX 消息队列的一个例程

5.1. 初始化 ThreadX 和消息队列

首先,需要在 STM32 项目中初始化 ThreadX 和创建消息队列。这通常在启动代码或初始化函数中进行。

#include "tx_api.h"  // 假设你已经初始化了 ThreadX 内核  
TX_THREAD           thread0;  
TX_THREAD           thread1;  
TX_QUEUE            my_queue;  void tx_initialize(void)  
{  // 初始化 ThreadX 内核(这通常是由 ThreadX 的启动函数完成的)  tx_kernel_enter();  // 创建消息队列  tx_queue_create(&my_queue, "My Queue", my_queue_buffer, sizeof(my_queue_buffer), TX_NO_WAIT);  // 创建线程(这里只是示例,你需要定义线程函数和堆栈)  tx_thread_create(&thread0, "Thread 0", thread0_entry, 0, thread0_stack, sizeof(thread0_stack),   16, 16, TX_NO_TIME_SLICE, TX_AUTO_START);  tx_thread_create(&thread1, "Thread 1", thread1_entry, 0, thread1_stack, sizeof(thread1_stack),   16, 16, TX_NO_TIME_SLICE, TX_AUTO_START);  
}  // 消息队列缓冲区  
UINT my_queue_buffer[1024];  // 线程堆栈(需要根据你的需要进行定义)  
UCHAR thread0_stack[STACK_SIZE];  
UCHAR thread1_stack[STACK_SIZE];  // ... 线程函数定义(thread0_entry, thread1_entry)...

5.2. 线程中使用消息队列

在线程函数中,你可以使用 tx_queue_send 和 tx_queue_receive 来发送和接收消息。

void thread0_entry(ULONG thread_input)  
{  UINT status;  ULONG send_message = 1234; // 要发送的消息  while(1)  {  // 发送消息到队列  status = tx_queue_send(&my_queue, &send_message, TX_WAIT_FOREVER);  if (status != TX_SUCCESS)  {  // 错误处理  }  // ... 其他处理 ...  tx_thread_sleep(100); // 休眠一段时间  }  
}  void thread1_entry(ULONG thread_input)  
{  UINT status;  ULONG received_message;  while(1)  {  // 从队列接收消息  status = tx_queue_receive(&my_queue, &received_message, TX_WAIT_FOREVER);  if (status != TX_SUCCESS)  {  // 错误处理  }  // 处理接收到的消息  // ...  tx_thread_sleep(100); // 休眠一段时间  }  
}

5.3. 注意事项

  • 确保你已经正确配置了 STM32 的硬件和启动代码,以支持 ThreadX。
  • 根据你的需要调整线程堆栈大小和优先级。
  • 错误处理在这里被简化了,你可能需要根据你的应用程序添加更详细的错误处理代码。
  • TX_WAIT_FOREVER 表示线程将无限期地等待,直到消息可用或发生错误。你也可以使用其他等待选项,如 TX_NO_WAIT 或指定一个超时值。

这篇关于threadX 消息队列的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java中Springboot集成Kafka实现消息发送和接收功能

《Java中Springboot集成Kafka实现消息发送和接收功能》Kafka是一个高吞吐量的分布式发布-订阅消息系统,主要用于处理大规模数据流,它由生产者、消费者、主题、分区和代理等组件构成,Ka... 目录一、Kafka 简介二、Kafka 功能三、POM依赖四、配置文件五、生产者六、消费者一、Kaf

Redis延迟队列的实现示例

《Redis延迟队列的实现示例》Redis延迟队列是一种使用Redis实现的消息队列,本文主要介绍了Redis延迟队列的实现示例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习... 目录一、什么是 Redis 延迟队列二、实现原理三、Java 代码示例四、注意事项五、使用 Redi

SpringBoot 自定义消息转换器使用详解

《SpringBoot自定义消息转换器使用详解》本文详细介绍了SpringBoot消息转换器的知识,并通过案例操作演示了如何进行自定义消息转换器的定制开发和使用,感兴趣的朋友一起看看吧... 目录一、前言二、SpringBoot 内容协商介绍2.1 什么是内容协商2.2 内容协商机制深入理解2.2.1 内容

hdu1180(广搜+优先队列)

此题要求最少到达目标点T的最短时间,所以我选择了广度优先搜索,并且要用到优先队列。 另外此题注意点较多,比如说可以在某个点停留,我wa了好多两次,就是因为忽略了这一点,然后参考了大神的思想,然后经过反复修改才AC的 这是我的代码 #include<iostream>#include<algorithm>#include<string>#include<stack>#include<

poj 3190 优先队列+贪心

题意: 有n头牛,分别给他们挤奶的时间。 然后每头牛挤奶的时候都要在一个stall里面,并且每个stall每次只能占用一头牛。 问最少需要多少个stall,并输出每头牛所在的stall。 e.g 样例: INPUT: 51 102 43 65 84 7 OUTPUT: 412324 HINT: Explanation of the s

poj 2431 poj 3253 优先队列的运用

poj 2431: 题意: 一条路起点为0, 终点为l。 卡车初始时在0点,并且有p升油,假设油箱无限大。 给n个加油站,每个加油站距离终点 l 距离为 x[i],可以加的油量为fuel[i]。 问最少加几次油可以到达终点,若不能到达,输出-1。 解析: 《挑战程序设计竞赛》: “在卡车开往终点的途中,只有在加油站才可以加油。但是,如果认为“在到达加油站i时,就获得了一

poj3750约瑟夫环,循环队列

Description 有N个小孩围成一圈,给他们从1开始依次编号,现指定从第W个开始报数,报到第S个时,该小孩出列,然后从下一个小孩开始报数,仍是报到S个出列,如此重复下去,直到所有的小孩都出列(总人数不足S个时将循环报数),求小孩出列的顺序。 Input 第一行输入小孩的人数N(N<=64) 接下来每行输入一个小孩的名字(人名不超过15个字符) 最后一行输入W,S (W < N),用

POJ2010 贪心优先队列

c头牛,需要选n头(奇数);学校总共有f的资金, 每头牛分数score和学费cost,问合法招生方案中,中间分数(即排名第(n+1)/2)最高的是多少。 n头牛按照先score后cost从小到大排序; 枚举中间score的牛,  预处理左边与右边的最小花费和。 预处理直接优先队列贪心 public class Main {public static voi

Java并发编程之——BlockingQueue(队列)

一、什么是BlockingQueue BlockingQueue即阻塞队列,从阻塞这个词可以看出,在某些情况下对阻塞队列的访问可能会造成阻塞。被阻塞的情况主要有如下两种: 1. 当队列满了的时候进行入队列操作2. 当队列空了的时候进行出队列操作123 因此,当一个线程试图对一个已经满了的队列进行入队列操作时,它将会被阻塞,除非有另一个线程做了出队列操作;同样,当一个线程试图对一个空

ActiveMQ—消息特性(延迟和定时消息投递)

ActiveMQ消息特性:延迟和定时消息投递(Delay and Schedule Message Delivery) 转自:http://blog.csdn.net/kimmking/article/details/8443872 有时候我们不希望消息马上被broker投递出去,而是想要消息60秒以后发给消费者,或者我们想让消息没隔一定时间投递一次,一共投递指定的次数。。。 类似