RT-Thread 消息队列测试

创建三个线程,1个发送优先级2,2个接受优先级为3、6,两个接受线程同时接受一个消息队列内容观察效果,代码简要:

/*
*************************************************************************
*                             包含的头文件
*************************************************************************
*/
#include "board.h"
#include "rtthread.h"


/*
*************************************************************************
*                               变量
*************************************************************************
*/
/* 定义线程控制块 */
static rt_thread_t receive_thread = RT_NULL;
static rt_thread_t receive2_thread = RT_NULL;
static rt_thread_t send_thread = RT_NULL;
/* 定义消息队列控制块 */
static rt_mq_t test_mq = RT_NULL;

/*
*************************************************************************
*                             函数声明
*************************************************************************
*/
static void receive_thread_entry(void* parameter);
static void send_thread_entry(void* parameter);

static void receive2_thread_entry(void* parameter);
/*
*************************************************************************
*                             main 函数
*************************************************************************
*/
/**
  * @brief  主函数
  * @param  无
  * @retval 无
  */
int main(void)
{
    /*
     * 开发板硬件初始化,RTT系统初始化已经在main函数之前完成,
     * 即在component.c文件中的rtthread_startup()函数中完成了。
     * 所以在main函数中,只需要创建线程和启动线程即可。
     */
    rt_kprintf("多线程接受一个消息队列\n");
    /* 创建一个消息队列 */
    test_mq = rt_mq_create("test_mq",			/* 消息队列名字 	  */
                           40,     				/* 消息的最大长度 	  */
                           20,    				/* 消息队列的最大容量 */
                           RT_IPC_FLAG_FIFO);	/* 队列模式 FIFO(0x00)*/
    if (test_mq != RT_NULL)
        rt_kprintf("消息队列1创建成功!\n\n");						   
						       


    receive_thread =                           /* 线程控制块指针 	  */
        rt_thread_create( "receive",           /* 线程名字 			  */
                          receive_thread_entry,/* 线程入口函数 		  */
                          RT_NULL,             /* 线程入口函数参数    */
                          256,                 /* 线程栈大小 		  */
                          3,                   /* 线程的优先级 		  */
                          20);                 /* 线程时间片 		  */

    /* 启动线程,开启调度 */
    if (receive_thread != RT_NULL)
        rt_thread_startup(receive_thread);
    else
        return -1;

    send_thread =                          	   /* 线程控制块指针	  */
        rt_thread_create( "send",              /* 线程名字 			  */
                          send_thread_entry,   /* 线程入口函数 		  */
                          RT_NULL,             /* 线程入口函数参数 	  */
                          256,                 /* 线程栈大小 		  */
                          2,                   /* 线程的优先级 		  */
                          20);                 /* 线程时间片 		  */

    /* 启动线程,开启调度 */
    if (send_thread != RT_NULL)
        rt_thread_startup(send_thread);
    else
        return -1;
		
	receive2_thread =                          	   /* 线程控制块指针	  */
        rt_thread_create( "receive2",              /* 线程名字 			  */
                          receive2_thread_entry,   /* 线程入口函数 		  */
                          RT_NULL,             /* 线程入口函数参数 	  */
                          256,                 /* 线程栈大小 		  */
                          6,                   /* 线程的优先级 		  */
                          20);                 /* 线程时间片 		  */

    /* 启动线程,开启调度 */
    if (receive2_thread != RT_NULL)
        rt_thread_startup(receive2_thread);
    else
        return -1;
}

/*
*************************************************************************
*                             线程定义
*************************************************************************
*/

static void receive_thread_entry(void* parameter)
{
    rt_err_t uwRet = RT_EOK;
    uint32_t r_queue;
    /* 任务都是一个无限循环,不能返回 */
    while (1)
    {
        /* 队列读取(接收),等待时间为一直等待 */
        uwRet = rt_mq_recv(test_mq,				 /* 读取(接收)队列的ID(句柄)	*/
                           &r_queue,			 /* 读取(接收)的数据保存位置	*/
                           sizeof(r_queue),		 /* 读取(接收)的数据的长度	*/
                           RT_WAITING_FOREVER);  /* 等待时间:一直等 			*/
        if(RT_EOK == uwRet)
        {
            rt_kprintf("本次接收到的数据是:%d\n",r_queue);
        }
        else
        {
            rt_kprintf("数据接收出错,错误代码: 0x%lx\n",uwRet);
        }
        rt_thread_delay(200);
    }
}

static void receive2_thread_entry(void* parameter)
{
    rt_err_t uwRet = RT_EOK;
    uint32_t r_queue2;
    /* 任务都是一个无限循环,不能返回 */
    while (1)
    {
        /* 队列读取(接收),等待时间为一直等待 */
        uwRet = rt_mq_recv(test_mq,				 /* 读取(接收)队列的ID(句柄)	*/
                           &r_queue2,			 /* 读取(接收)的数据保存位置	*/
                           sizeof(r_queue2),		 /* 读取(接收)的数据的长度	*/
                           RT_WAITING_FOREVER);  /* 等待时间:一直等 			*/
        if(RT_EOK == uwRet)
        {
            rt_kprintf("本次接收到的数据是2:%d\n",r_queue2);
        }
        else
        {
            rt_kprintf("数据接收出错,错误代码2: 0x%lx\n",uwRet);
        }
        rt_thread_delay(200);
    }
}

static void send_thread_entry(void* parameter)
{
    rt_err_t uwRet = RT_EOK;
    uint32_t send_data1 = 1;
    uint32_t send_data2 = 2;
    while (1)
    {
        if( Key_Scan(KEY1_GPIO_PORT,KEY1_GPIO_PIN) == KEY_ON )/* K1 被按下 */
        {
            /* 将数据写入(发送)到队列中,等待时间为 0  */
            uwRet = rt_mq_send(	test_mq,	/* 写入(发送)队列的ID(句柄) */
                                &send_data1,			/* 写入(发送)的数据 */
                                sizeof(send_data1));			/* 数据的长度 */
													
            if(RT_EOK != uwRet)
            {
                rt_kprintf("数据不能发送到消息队列1!错误代码: %lx\n",uwRet);
            }
			
							
								
								
        }
        if( Key_Scan(KEY2_GPIO_PORT,KEY2_GPIO_PIN) == KEY_ON )/* K1 被按下 */
        {
            /* 将数据写入(发送)到队列中,等待时间为 0  */
            uwRet = rt_mq_send(	test_mq,	/* 写入(发送)队列的ID(句柄) */
                                &send_data2,			/* 写入(发送)的数据 */
                                sizeof(send_data2));			/* 数据的长度 */
            if(RT_EOK != uwRet)
            {
                rt_kprintf("数据不能发送到消息队列!错误代码: %lx\n",uwRet);
            }
        }
        rt_thread_delay(20);
    }
}




/********************************END OF FILE****************************/

多线程同时接受一个消息队列内容出现的现象

两个线不能同时接受内容,因为在接受消息队列函数中接受完消息就会将消息删除

 

    /* get message from queue */
    msg = (struct rt_mq_message *)mq->msg_queue_head;

    /* move message queue head */
    mq->msg_queue_head = msg->next;
    /* reach queue tail, set to NULL */
    if (mq->msg_queue_tail == msg)
        mq->msg_queue_tail = RT_NULL;

    /* decrease message entry */
    mq->entry --;

    /* enable interrupt */
    rt_hw_interrupt_enable(temp);

    /* copy message */
    rt_memcpy(buffer, msg + 1, size > mq->msg_size ? mq->msg_size : size);

    /* disable interrupt */
    temp = rt_hw_interrupt_disable();
    /* put message to free list */
    msg->next = (struct rt_mq_message *)mq->msg_queue_free;
    mq->msg_queue_free = msg;
    /* enable interrupt */
    rt_hw_interrupt_enable(temp);

    RT_OBJECT_HOOK_CALL(rt_object_take_hook, (&(mq->parent.parent)));

    return RT_EOK;

 为避免这个问题我们使用两个不同的消息队列观察效果

 

 简要代码



/*
*************************************************************************
*                             包含的头文件
*************************************************************************
*/
#include "board.h"
#include "rtthread.h"


/*
*************************************************************************
*                               变量
*************************************************************************
*/
/* 定义线程控制块 */
static rt_thread_t receive_thread = RT_NULL;
static rt_thread_t receive2_thread = RT_NULL;
static rt_thread_t send_thread = RT_NULL;
/* 定义消息队列控制块 */
static rt_mq_t test_mq = RT_NULL;
static rt_mq_t tes2_mq = RT_NULL;
/*
*************************************************************************
*                             函数声明
*************************************************************************
*/
static void receive_thread_entry(void* parameter);
static void send_thread_entry(void* parameter);

static void receive2_thread_entry(void* parameter);
/*
*************************************************************************
*                             main 函数
*************************************************************************
*/
/**
  * @brief  主函数
  * @param  无
  * @retval 无
  */
int main(void)
{
    /*
     * 开发板硬件初始化,RTT系统初始化已经在main函数之前完成,
     * 即在component.c文件中的rtthread_startup()函数中完成了。
     * 所以在main函数中,只需要创建线程和启动线程即可。
     */
    rt_kprintf("多线程接受一个消息队列\n");
    /* 创建一个消息队列 */
    test_mq = rt_mq_create("test_mq",			/* 消息队列名字 	  */
                           40,     				/* 消息的最大长度 	  */
                           20,    				/* 消息队列的最大容量 */
                           RT_IPC_FLAG_FIFO);	/* 队列模式 FIFO(0x00)*/
    if (test_mq != RT_NULL)
        rt_kprintf("消息队列1创建成功!\n\n");						   
						       
    tes2_mq = rt_mq_create("tes2_mq",			/* 消息队列名字 	  */
                           40,     				/* 消息的最大长度 	  */
                           20,    				/* 消息队列的最大容量 */
                           RT_IPC_FLAG_FIFO);	/* 队列模式 FIFO(0x00)*/
    if (tes2_mq != RT_NULL)
        rt_kprintf("消息队列2创建成功!\n\n");

    receive_thread =                           /* 线程控制块指针 	  */
        rt_thread_create( "receive",           /* 线程名字 			  */
                          receive_thread_entry,/* 线程入口函数 		  */
                          RT_NULL,             /* 线程入口函数参数    */
                          256,                 /* 线程栈大小 		  */
                          3,                   /* 线程的优先级 		  */
                          20);                 /* 线程时间片 		  */

    /* 启动线程,开启调度 */
    if (receive_thread != RT_NULL)
        rt_thread_startup(receive_thread);
    else
        return -1;

    send_thread =                          	   /* 线程控制块指针	  */
        rt_thread_create( "send",              /* 线程名字 			  */
                          send_thread_entry,   /* 线程入口函数 		  */
                          RT_NULL,             /* 线程入口函数参数 	  */
                          256,                 /* 线程栈大小 		  */
                          2,                   /* 线程的优先级 		  */
                          20);                 /* 线程时间片 		  */

    /* 启动线程,开启调度 */
    if (send_thread != RT_NULL)
        rt_thread_startup(send_thread);
    else
        return -1;
		
	receive2_thread =                          	   /* 线程控制块指针	  */
        rt_thread_create( "receive2",              /* 线程名字 			  */
                          receive2_thread_entry,   /* 线程入口函数 		  */
                          RT_NULL,             /* 线程入口函数参数 	  */
                          256,                 /* 线程栈大小 		  */
                          6,                   /* 线程的优先级 		  */
                          20);                 /* 线程时间片 		  */

    /* 启动线程,开启调度 */
    if (receive2_thread != RT_NULL)
        rt_thread_startup(receive2_thread);
    else
        return -1;
}

/*
*************************************************************************
*                             线程定义
*************************************************************************
*/

static void receive_thread_entry(void* parameter)
{
    rt_err_t uwRet = RT_EOK;
    uint32_t r_queue;
    /* 任务都是一个无限循环,不能返回 */
    while (1)
    {
        /* 队列读取(接收),等待时间为一直等待 */
        uwRet = rt_mq_recv(test_mq,				 /* 读取(接收)队列的ID(句柄)	*/
                           &r_queue,			 /* 读取(接收)的数据保存位置	*/
                           sizeof(r_queue),		 /* 读取(接收)的数据的长度	*/
                           RT_WAITING_FOREVER);  /* 等待时间:一直等 			*/
        if(RT_EOK == uwRet)
        {
            rt_kprintf("本次接收到的数据是:%d\n",r_queue);
        }
        else
        {
            rt_kprintf("数据接收出错,错误代码: 0x%lx\n",uwRet);
        }
        rt_thread_delay(200);
    }
}

static void receive2_thread_entry(void* parameter)
{
    rt_err_t uwRet = RT_EOK;
    uint32_t r_queue2;
    /* 任务都是一个无限循环,不能返回 */
    while (1)
    {
        /* 队列读取(接收),等待时间为一直等待 */
        uwRet = rt_mq_recv(tes2_mq,				 /* 读取(接收)队列的ID(句柄)	*/
                           &r_queue2,			 /* 读取(接收)的数据保存位置	*/
                           sizeof(r_queue2),		 /* 读取(接收)的数据的长度	*/
                           RT_WAITING_FOREVER);  /* 等待时间:一直等 			*/
        if(RT_EOK == uwRet)
        {
            rt_kprintf("本次接收到的数据是2:%d\n",r_queue2);
        }
        else
        {
            rt_kprintf("数据接收出错,错误代码2: 0x%lx\n",uwRet);
        }
        rt_thread_delay(200);
    }
}

static void send_thread_entry(void* parameter)
{
    rt_err_t uwRet = RT_EOK;
	rt_err_t u2Ret = RT_EOK;
    uint32_t send_data1 = 1;
    uint32_t send_data2 = 2;
    while (1)
    {
        if( Key_Scan(KEY1_GPIO_PORT,KEY1_GPIO_PIN) == KEY_ON )/* K1 被按下 */
        {
            /* 将数据写入(发送)到队列中,等待时间为 0  */
            uwRet = rt_mq_send(	test_mq,	/* 写入(发送)队列的ID(句柄) */
                                &send_data1,			/* 写入(发送)的数据 */
                                sizeof(send_data1));			/* 数据的长度 */
													
            if(RT_EOK != uwRet)
            {
                rt_kprintf("数据不能发送到消息队列1!错误代码: %lx\n",uwRet);
            }
			
			u2Ret= rt_mq_send(	tes2_mq,	/* 写入(发送)队列的ID(句柄) */
                                &send_data1,			/* 写入(发送)的数据 */
                                sizeof(send_data1));			/* 数据的长度 */
            if(RT_EOK != u2Ret)
            {
                rt_kprintf("数据不能发送到消息队列2!错误代码: %lx\n",uwRet);
            }								
								
								
        }
        if( Key_Scan(KEY2_GPIO_PORT,KEY2_GPIO_PIN) == KEY_ON )/* K1 被按下 */
        {
            /* 将数据写入(发送)到队列中,等待时间为 0  */
            uwRet = rt_mq_send(	test_mq,	/* 写入(发送)队列的ID(句柄) */
                                &send_data2,			/* 写入(发送)的数据 */
                                sizeof(send_data2));			/* 数据的长度 */
            if(RT_EOK != uwRet)
            {
                rt_kprintf("数据不能发送到消息队列!错误代码: %lx\n",uwRet);
            }
			
			
			u2Ret = rt_mq_send(	tes2_mq,	/* 写入(发送)队列的ID(句柄) */
                                &send_data2,			/* 写入(发送)的数据 */
                                sizeof(send_data2));			/* 数据的长度 */
            if(RT_EOK != u2Ret)
            {
                rt_kprintf("数据不能发送到消息队列!错误代码: %lx\n",uwRet);
            }
        }
        rt_thread_delay(20);
    }
}




/********************************END OF FILE****************************/

  • 0
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

Steven&Aileen

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

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

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

打赏作者

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

抵扣说明:

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

余额充值