5.1 概述

消息队列可以认为是一个链表。有写权限的线程可往消息队列中放置消息,有读权限的线程可以从消息队列中取走消息。

消息队列和管道/FIFO的区别:

(1)消息队列往一个队列中写消息前,并不需要有一个进程/线程在等待消息的到达,管道/FIFO需要。

(2)Posix消息队列是随内核的持续性,管道/FIFO是随进程的。

Posix消息队列和System V消息队列的区别:

(1)Posix消息队列的读总是返回最高优先级(类型)的最早消息;System V消息队列的读可以返回指定优先级(类型)的消息;

(2)Posix消息队列可以产生一个信号,System V消息队列不可以。

5.2 函数

5.2.1 mq_open函数

头文件 #include <mqueue.h>
函数原型 mqd_t mq_open(const char *name, int oflag);
  mqd_t mq_open(const char *name, int oflag, mode_t mode, struct mq_attr *attr);
返回值 成功返回消息队列的描述符,失败返回-1
  name:只能以/开头,且之后不能有/
参数 oflag:O_RDONLY、O_WRONLY、O_RDWR、O_CREAT、O_EXCL、O_NOBLOCK
  mode:0666
说明 1.有2个函数的原因和open函数一样,参数中有...可变参数
  2.第4个参数,有的版本传变量的地址,有的版本传变量

5.2.2 mq_close函数

头文件 #include <mqueue.h>
函数原型 int mq_close(mqd_t mqdes);
返回值 成功返回0,失败返回-1
说明 只是关闭消息队列描述符,但消息队列还在系统中

5.2.3 mq_unlink 函数

头文件 #include <mqueue.h>
函数原型 int mq_unlink(const char *name);
返回值 成功返回0,失败返回-1
说明 删除消息队列

注意:

1.编译时需要加上 -lrt

2.程序编译完成后,如果直接运行程序则提示mq_open失败,提示mq_open permission denied。解决办法是:

mkdir /dev/mqueue
mount -t mqueue none /dev/mqueue

5.2.4 mqcreat 程序

 #include <stdio.h>
 #include <unistd.h>
 #include <mqueue.h>

 // mqcreat -e /mqueue
 int main(int argc, char **argv)
 {
     ;
     int flags = O_RDWR | O_CREAT;
     mqd_t mqd = ;

     )
     {
         switch (c)
         {
             case 'e':
                 flags |= O_EXCL;
                 break;
             default:
                 break;
         }
     }

     )
     {
         printf("usage: mqcreate [-e] <name>\n");
         ;
     }

     // 打开或创建一个消息队列
     mqd = mq_open(argv[optind], flags, , );
     )
     {
         perror("mq_open error");
         ;
     }

     // 关闭消息队列
     mq_close(mqd);

     // 删除消息队列
     //mq_unlink(argv[optind]);

     ;
 }

5.3 消息队列的属性

struct mq_attr {
  long mq_flags; /* Flags: 0 or O_NONBLOCK */
  long mq_maxmsg; /* Max. # of messages on queue */
  long mq_msgsize; /* Max. message size (bytes) */
  long mq_curmsgs; /* # of messages currently in queue */
};

每个消息队列有4个属性:是否阻塞、允许最大的消息数量、允许最大的字节、当前消息数量。

1.如果创建一个新队列,指定了mq_maxmsg、mq_msgsize则忽略其他两个成员。

2.如果设置成非阻塞的,则另外三个参数被忽略。

头文件 #include <mqueue.h>
函数原型 int mq_getattr(mqd_t mqdes, struct mq_attr *attr);
  int mq_setattr(mqd_t mqdes, const struct mq_attr *attr, struct mq_attr *oattr);
返回值 成功返回0,失败返回-1

5.3.1 带属性的mq_creat程序

 #include <stdio.h>
 #include <unistd.h>
 #include <string.h>
 #include <mqueue.h>

 // mqcreat -e -m 1024 -z 8192 /mqueue
 int main(int argc, char **argv)
 {
     ;
     int flags = O_RDWR | O_CREAT;
     mqd_t mqd = ;
     struct mq_attr attr;
     memset(&attr, , sizeof(attr));

     )
     {
         switch (c)
         {
             case 'e':
                 flags |= O_EXCL;
                 break;
             case 'm':
                 attr.mq_maxmsg = atol(optarg);
                 break;
             case 'z':
                 attr.mq_msgsize = atol(optarg);
                 break;
             default:
                 break;
         }
     }

     )
     {
         printf("usage: mqcreate [-e] <name>\n");
         ;
     }

     // 打开或创建一个消息队列
     mqd = mq_open(argv[optind], flags, , attr);
     )
     {
         perror("mq_open error");
         ;
     }

     // 关闭消息队列
     mq_close(mqd);

     // 删除消息队列
     mq_unlink(argv[optind]);

     ;
 }

5.4 mq_send 和 mq_receive 函数

  mq_send是往消息队列中放消息;mq_receive从消息队列中取消息。每个消息都有一个优先级(类型),它是一个小于MQ_PRIO_MAX的无符号整数,Posxi的上限是32。

  mq_receive总是返回队列中在最高优先级的最早消息。而System V的msgrcv函数可以返回指定优先级的最早消息。

头文件 #include <mqueue.h>
函数原型 int mq_send(mqd_t mqdes, const char *msg_ptr, size_t msg_len, unsigned int msg_prio);
返回值 成功返回0,失败返回-1
函数原型 ssize_t mq_receive(mqd_t mqdes, char *msg_ptr, size_t msg_len, unsigned int *msg_prio);
返回值 成功返回字节数,失败返回-1
说明 1.mq_receive的len参数不能小于消息的最大值(mq_attr结构中mq_msgsize成员),如果小于就立即返回EMSGSIZE错误
  2.mq_send的prio参数是待发消息的优先级(类型),其值必须小于32位无符号整数
  3.mq_receive的prio参数如果非空,所返回消息的优先级(类型)就放在这个地址中
  4.如果不需要指定优先级,则mq_send的prio参数为0,mq_receive的prio参数为NULL 

5.4.1 mqsend程序

 #include <stdio.h>
 #include <stdlib.h>
 #include <unistd.h>
 #include <string.h>
 #include <mqueue.h>

 // mqsend /mqueue 100 1
 int main(int argc, char **argv)
 {
     ;
     int flags = O_RDWR | O_CREAT;
     mqd_t mqd = ;
     struct mq_attr attr;
     memset(&attr, , sizeof(attr));

     )
     {
         printf("usage: mqsend <name> <bytes> <priority>\n");
         ;
     }

     // 打开或创建一个消息队列
     mqd = mq_open(argv[], flags, , NULL);
     )
     {
         perror("mq_open error");
         ;
     }

     unsigned ]);
     unsigned ]);
     , len);

     int ret = mq_send(mqd, ptr, len, prio);
     )
     {
         perror("mq_send error");
         ;
     }

     // 关闭消息队列
     mq_close(mqd);

     ;
 }

5.4.2 mqreceive程序

  命令行选项-n指定非阻塞模式,这样消息队列中没有消息,mq_receive就返回一个错误。

 #include <stdio.h>
 #include <stdlib.h>
 #include <unistd.h>
 #include <string.h>
 #include <mqueue.h>

 // mqreceive [-n] /mqueue
 int main(int argc, char **argv)
 {
     ;
     int flag = O_RDWR | O_CREAT;
     mqd_t mqd = ;
     struct mq_attr attr;
     unsigned ;

     )
     {
         switch (c)
         {
             case 'n':
                 flag |= O_NONBLOCK;
                 break;
             default:
                 break;
         }
     }

     )
     {
         printf("usage: mareceive [-n] <name>\n");
         ;
     }

     // 打开消息队列
     mqd = mq_open(argv[optind], flag, , );
     )
     {
         perror("mq_open error");
         ;
     }

     // 获取消息队列中消息的最大字节数
     mq_getattr(mqd, &attr);

     );

     int n = mq_receive(mqd, buff, attr.mq_msgsize, &prio);
     printf("read %d bytes, prio:%d\n", n, prio);

     ;
 }

5.5 消息队列限制

  mq_maxmsg  队列中的最大消息数,没有限制

  mq_msgsize  消息的最大字节数,没有限制

  MQ_OPEN_MAX  一个进程能同时打开消息队列的最大数目,有限制,查询方法是:sysconf(_SC_MQ_OPEN_MAX);

  MQ_PRIO_MAX  消息的优先级要小于这个值,有限制,查询方法是:sysconf(_SC_MQ_PRIO_MAX);

5.5.1 mqsysconf 程序

第5章 Posix 消息队列的更多相关文章

  1. 第三十四章 POSIX消息队列

    POSIX消息队列相关函数 mq_open 功能: 用来创建和访问一个消息队列 原型: mqd_t mq_open(const char *name, int oflag); //只能用来打开消息队列 ...

  2. Linux IPC POSIX 消息队列

    模型: #include<mqueue.h> #include <sys/stat.h> #include <fcntl.h> mq_open() //创建/获取消 ...

  3. Posix消息队列

    转载于:http://blog.csdn.net/zx714311728/article/details/53197196 1.消息队列 消息队列可以认为是一个消息链表,消息队列是随内核持续的.队列中 ...

  4. posix 消息队列

    注意 在涉及到posix消息的函数时, gcc 编译时要加-lrt参数, 如 gcc -lrt unpipc.c mqpack.c send.c -o send gcc -lrt unpipc.c m ...

  5. Posix消息队列实现机制

    本文是对<Unix 网络编程 卷2:进程通信>的笔记. 引言 消息队列是进程间通信的一种方式,可是如果不理解他的实现原理,会有众多不理解之处,下面就结合本书中的例子,对posix消息队列来 ...

  6. POSIX 消息队列相关问题

    一.查看和删除消息队列要想看到创建的posix消息队列,需要在root用户下执行以下操作:# mkdir /dev/mqueue# mount -t mqueue none /dev/mqueue删除 ...

  7. Linux IPC实践(7) --Posix消息队列

    1. 创建/获取一个消息队列 #include <fcntl.h> /* For O_* constants */ #include <sys/stat.h> /* For m ...

  8. 进程间通信--POSIX消息队列

    相关函数: mqd_t mq_open(const char *name, int oflag); mqd_t mq_send(mqd_t mqdes, const char *msg_ptr, si ...

  9. POSIX 消息队列 之 概述 链接方式

    NAMEmq_overview —— POSIX消息队列概述 DESCRIPTIONPOSIX消息队列允许进程以消息的形式交换数据.此API与System V消息队列(msgget(2),msgsnd ...

随机推荐

  1. log4j 使用笔记整理中

    Log4j由三个重要的组件构成:日志信息的优先级,日志信息的输出目的地,日志信息的输出格式. 日志信息的常用的优先级从高到低有ERROR.WARN. INFO.DEBUG,分别用来指定这条日志信息的重 ...

  2. 如何用Transformer+从PDF文档编辑数据

    ABBYY PDF Transformer+是一款可创建.编辑.添加注释及将PDF文件转换为其他可编辑格式的通用工具,可使用该软件从PDF文档编辑机密信息,然后再发布它们,文本和图像均可编辑,本文将为 ...

  3. 设置EDIUS字幕时有哪些要注意的

    我们在用EDIUS添加字幕,有时候可能会遇到以下麻烦.例如有的字体在EDIUS中找不到,诗歌的排版问题还有怎么给字幕加光效等等.今天小编主要来给大家解决这三个问题,让你们知道EDIUS字幕设置时应该注 ...

  4. SmartAdmin 打开速度慢的原因

    最近在使用SmartAdmin做个小东西,发布在公网上,我的机器打开飞快,但是到了其它人的机器上变得极慢了.而且在我的手机上也打开变慢.     查找原因,原来如此.      <link re ...

  5. cache与负载均衡

    cache 客户端CACHE客户端CACHE,包括浏览器本身的缓存.FLASH存储等,用于存储一些临时的文件或者变化不大或无变化的数据:1.如浏览器自动将用户浏览的网页存储在用户的硬盘上,下次再浏览相 ...

  6. #linux包之sysstat之iostat命令

    概述 对于I/O-bond类型的进程,我们经常用iostat工具查看进程IO请求下发的数量.系统处理IO请求的耗时,进而分析进程与操作系统的交互过程中IO方面是否存在瓶颈.同vmstat一样,iost ...

  7. unity, 用unity profiler进行真机profile,需要退出360

    用unity profiler进行真机profile,需要退出360.

  8. OpenJudge计算概论-字符串最大跨距

    /*====================================================================== 字符串最大跨距 总时间限制: 1000ms 内存限制: ...

  9. JQuery Pagenation 知识点整理——phototype 应用(20150517)(转)

    JS中的phototype是JS中比较难理解的一个部分 本文基于下面几个知识点: 1 原型法设计模式 在.Net中可以使用clone()来实现原型法 原型法的主要思想是,现在有1个类A,我想要创建一个 ...

  10. mysql 修改字符集

    查看mysql数据编码 登录MySQL服务,查看mysql数据库默认的编码 mysql> show variables like 'character%'; +----------------- ...