大家好,我是LinuxZn。
在应用开发中,生产者,消费者的模型非常常见,一方产生数据并把数据放入队列中,而另一方从队列中取数据,先进先出。
应用:线程间通信/进程间通信。Hello系列 | 多线程编程基础!
Linux系统中提供了两种不同接口的消息队列:
System V消息队列。System V 是 AT&T 的第一个商业UNIX版本(UNIX System III)的加强。
其中,POSIX消息队列可移植性较强,使用较广。
Linux系统中提供的消息队列一般应用于进行间通信,但也可以用于线程间通信。
本文介绍POSIX消息队列应用于线程间通信。
头文件:
#include/*ForO_*constants*/ #include /*Formodeconstants*/ #include
编译链接需要加上 -lr 链接。
Linux内核提供了一系列函数来使用消息队列:
/** *@brief创建消息队列实例 * *Detailedfunctiondescription * *@param[in]name:消息队列名称 *@param[in] oflag:根据传入标识来创建或者打开一个已创建的消息队列 -O_CREAT:创建一个消息队列 -O_EXCL:检查消息队列是否存在,一般与O_CREAT一起使用 -O_CREAT|O_EXCL:消息队列不存在则创建,已存在返回NULL -O_NONBLOCK:非阻塞模式打开,消息队列不存在返回NULL -O_RDONLY:只读模式打开 -O_WRONLY:只写模式打开 -O_RDWR:读写模式打开 *@param[in] mode:访问权限 *@param[in] attr:消息队列属性地址 * *@return成功返回消息队列描述符,失败返回-1,错误码存于error中 */ mqd_tmq_open(constchar*name,intoflag,mode_tmode,structmq_attr*attr); /** *@brief无限阻塞方式接收消息 * *Detailedfunctiondescription * *@param[in]mqdes:消息队列描述符 *@param[in] msg_ptr:消息体缓冲区地址 *@param[in] msg_len:消息体长度,长度必须大于等于消息属性设定的最大值 *@param[in] msg_prio:消息优先级 * *@return成功返回消息长度,失败返回-1,错误码存于error中 */ mqd_tmq_receive(mqd_tmqdes,char*msg_ptr,size_tmsg_len,unsigned*msg_prio); /** *@brief指定超时时间阻塞方式接收消息 * *Detailedfunctiondescription * *@param[in]mqdes:消息队列描述符 *@param[in] msg_ptr:消息体缓冲区地址 *@param[in] msg_len:消息体长度,长度必须大于等于消息属性设定的最大值 *@param[in] msg_prio:消息优先级 *@param[in] abs_timeout:超时时间 * *@return成功返回消息长度,失败返回-1,错误码存于error中 */ mqd_tmq_timedreceive(mqd_tmqdes,char*msg_ptr,size_tmsg_len,unsigned*msg_prio,conststructtimespec*abs_timeout); /** *@brief无限阻塞方式发送消息 * *Detailedfunctiondescription * *@param[in]mqdes:消息队列描述符 *@param[in] msg_ptr:待发送消息体缓冲区地址 *@param[in] msg_len:消息体长度 *@param[in] msg_prio:消息优先级 * *@return成功返回0,失败返回-1 */ mqd_tmq_send(mqd_tmqdes,constchar*msg_ptr,size_tmsg_len,unsignedmsg_prio); /** *@brief指定超时时间阻塞方式发送消息 * *Detailedfunctiondescription * *@param[in]mqdes:消息队列描述符 *@param[in] msg_ptr:待发送消息体缓冲区地址 *@param[in] msg_len:消息体长度 *@param[in] msg_prio:消息优先级 *@param[in] abs_timeout:超时时间 * *@return成功返回0,失败返回-1 */ mqd_tmq_timedsend(mqd_tmqdes,constchar*msg_ptr,size_tmsg_len,unsignedmsg_prio,conststructtimespec*abs_timeout); /** *@brief关闭消息队列 * *Detailedfunctiondescription * *@param[in]mqdes:消息队列描述符 * *@return成功返回0,失败返回-1 */ mqd_tmq_close(mqd_tmqdes); /** *@brief分离消息队列 * *Detailedfunctiondescription * *@param[in]name:消息队列名称 * *@return成功返回0,失败返回-1 */ mqd_tmq_unlink(constchar*name);
例子:线程1不断给线程2发送字符串数据。
#include#include #include #include #include #include /*ForO_*constants*/ #include /*Formodeconstants*/ #include #defineMQ_MSG_MAX_SIZE512///< 最大消息长度 #define MQ_MSG_MAX_ITEM 5 ///< 最大消息数目 static pthread_t s_thread1_id; static pthread_t s_thread2_id; static unsigned char s_thread1_running = 0; static unsigned char s_thread2_running = 0; static mqd_t s_mq; static char send_msg[10] = "hello"; void *thread1_fun(void *arg) { int ret = 0; s_thread1_running = 1; while (s_thread1_running) { ret = mq_send(s_mq, send_msg, sizeof(send_msg), 0); if (ret < 0) { perror("mq_send error"); } printf("send msg = %s ", send_msg); usleep(100 * 1000); } pthread_exit(NULL); } void *thread2_fun(void *arg) { char buf[MQ_MSG_MAX_SIZE]; int recv_size = 0; s_thread2_running = 1; while (s_thread2_running) { recv_size = mq_receive(s_mq, &buf[0], sizeof(buf), NULL); if (-1 != recv_size) { printf("receive msg = %s ", buf); } else { perror("mq_receive error"); break; } usleep(100 * 1000); } pthread_exit(NULL); } int main(void) { int ret = 0; struct mq_attr attr; ///< 创建消息队列 memset(&attr, 0, sizeof(attr)); attr.mq_maxmsg = MQ_MSG_MAX_ITEM; attr.mq_msgsize = MQ_MSG_MAX_SIZE; attr.mq_flags = 0; s_mq = mq_open("/mq", O_CREAT|O_RDWR, 0777, &attr); if(-1 == s_mq) { perror("mq_open error"); return -1; } ///< 创建线程1 ret = pthread_create(&s_thread1_id, NULL, thread1_fun, NULL); if (ret != 0) { printf("thread1_create error! "); exit(EXIT_FAILURE); } ret = pthread_detach(s_thread1_id); if (ret != 0) { printf("s_thread1_id error! "); exit(EXIT_FAILURE); } ///< 创建线程2 ret = pthread_create(&s_thread2_id, NULL, thread2_fun, NULL); if (ret != 0) { printf("thread2_create error! "); exit(EXIT_FAILURE); } ret = pthread_detach(s_thread2_id); if (ret != 0) { printf("s_thread2_id error! "); exit(EXIT_FAILURE); } while (1) { sleep(1); } return 0; }
编译、运行:
以上就是本次的分享,如果文章有帮助,麻烦帮忙转发,谢谢!
审核编辑:汤梓红
-
通信
+关注
关注
18文章
6021浏览量
135947 -
Linux
+关注
关注
87文章
11285浏览量
209268 -
Linux系统
+关注
关注
4文章
593浏览量
27387 -
线程
+关注
关注
0文章
504浏览量
19674 -
消息队列
+关注
关注
0文章
33浏览量
2972
原文标题:消息队列应用于线程间通信 | 简单例子
文章出处:【微信号:Linux大陆,微信公众号:Linux大陆】欢迎添加关注!文章转载请注明出处。
发布评论请先 登录
相关推荐
评论