消息队列和共享内存、信号量一样,同属 System V IPC 通信机制。消息队列是一系列连续排列的消息,保存在内核中,通过消息队列的引用标识符来访问。使用消息队列的好处是对每个消息指定了特定消息类型,接收消息的进程可以请求接收下一条消息,也可以请求接收下一条特定类型的消息。

相关数据结构

与其他两个 System V IPC 通信机制一样,消息队列也有一个与之对应的结构,该结构的定义如下:

struct msqid_ds
{
struct ipc_perm msq_perm;
struct msg *msg_first;
struct msg *msg_last;
ulong msg_ctypes;
ulong msg_qnum;
ulong msg_qbytes;
pid_t msg_lspid;
pid_t msg_lrpid;
time_t msg_stime;
time_t msg_rtime;
time_t msg_ctime;
}

该结构中各个字段的说明如下。
msg_perm:对应于该消息队列的 ipc_perm 结构指针。
msg_first:msg 结构指针,msg 结构用于表示一个消息,此指针指向消息队列中的第一个消息。
msg_last:msg 结构指针,指向消息队列中的最后一个消息。
msg_ctypes:记录消息队列中当前的总字节数。
msg_qnum:记录消息队列中当前的总消息数。
msg_qbytes:记录消息队列中最大可容纳的字节数。
msg_lspid:最近一个执行 msgsnd 函数的进程的 PID。
msg_lrpid:最近一个执行 msgrcv 函数的进程的 PID。
msg_stime:最近一次执行 msgsnd 函数的时间。
msg_rtime:最近一次执行 msgrcv 函数的时间。
msg_ctime:最近一次改变该消息队列的时间。
消息队列所传递的消息由两部分组成,即消息的类型及所传递的数据。一般用一个结构体来表示。通常消息类型用一个正的长整数表示,而数据则根据需要设定。比如设定一个传递 1024 个字节长度的字符串数据的消息如下:

struct msgbuf
{
long msgtype;
char msgtext[];
}

传递消息时将所传递的数据内容写入 msgtext 中,然后把这个结构体发送到消息队列中即可。

消息队列相关的函数

消息队列的创建与打开
要使用消息队列,首先要创建一个消息队列,创建消息队列的函数声明如下:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h> int msgget(key_t key, int msgflg);

函数 msgget 用于创建或打开一个消息队列。其中,参数 key 表示所创建或打开的消息队列的键。参数 msgflg 表示调用函数的操作类型,也可用于设置消息队列的访问权限,两者通过逻辑或表示。调用函数 msgget 所执行的具体操作由参数 key 和 flag 决定。相应约定与 shmget 函数类似。
函数调用成功时,返回值为消息队列的引用标识符。调用失败时,返回值为 -1。
当调用 msgget 函数创建一个消息队列时,它相应的 msqid_ds 结构被初始化。Ipc_perm 中各个字段被设置为相应的值,其中 msg_qnum、msg_lspid、msg_lrpid、msg_stime 和 msg_rt 都被设置为 0,msg_qtypes 被设置为系统限制值,msg_ctime 被设置为当前时间。

向消息队列中发送消息
接下来我们介绍如何向一个消息队列中发送消息,发送消息的函数声明如下:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h> int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);

此函数的作用是向一个消息队列中发送消息。该消息将被添加到消息队列的末尾。参数 msqid 是消息队列的引用标识符。参数 msgp 是一个 void 指针,指向要发送的消息。参数 msgsz 是以字节数标识的消息数据的长度。参数 msgflg 用于指定消息队列充满时的处理方法。当消息队列充满时,如果设置了 IPC_NOWAIT 位,就立即出错返回,否则发送消息的进程被阻塞,直至消息队列中有空间或该消息队列被删除时,函数返回。
msgsnd 函数调用成功时,返回值为 0,调用失败时,返回值为 -1。

从消息队列中接收消息
进程要从消息队列中接收消息时,需要调用 msgrcv 函数,其声明如下:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h> ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);

此函数用于从指定的消息队列中接收消息。参数 msqid 是消息队列的引用标识符。参数 msgp 是一个 void 指针,接收到的消息将被存放在 msgp 所指向的缓冲区。参数 msgsz 是以字节数表示的要接收的消息的长度。当消息的实际长度大于这个值时,将根据 msgflg 的设置做出相应的处理。参数 msgtyp 用于表示要接收的消息的类型,其取值和含义如下:
msgtyp=0    接收消息队列中的第一条消息
msgtyp>0    接收消息队列中类型值等于 msgtyp 的第一条消息
msgtyp<0    接收消息队列中类型值小于等于 msgtyp 的绝对值的所有消息类型值最小的消息中的第一条消息
参数 msgflg 用于设定与接收消息相关的信息。
IPC_NOWAIT:指定 msgtyp 无效时的处理方法。当 msgtyp 无效时,如果 IPC_NOWAIT 被设置,则立即出错返回,否则接收消息的进程将被阻塞,直至 msgtyp 有效或该消息队列被删除。
MSG_NOERROR:用于设置消息长度大于 msgsz 时的处理方法。当消息长度大于 msgsz 时,如果 MSG_NOERROR 位被设置,则接收该消息,超出部分被截断,函数正确返回,否则不接收该消息而将其保留在消息队列中,出错返回。
函数 msgrcv 调用成功时,返回值为以字节数表示的接收到的消息数据的长度,调用失败时,返回值为 -1。

消息队列的控制
对消息队列的具体控制操作是通过函数 msgctl 来实现的,其声明如下:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h> int msgctl(int msqid, int cmd, struct msqid_ds *buf);

其中,参数 msqid 为消息队列的引用标识符。参数 cmd 表示调用该函数希望执行的操作,其取值和相关说明如下。
IPC_RMID:删除消息队列。此命令是立即执行的,如果还有进程对此消息队列进行操作,则出错返回。只有有效用户 ID 和消息队列的所有者 ID 或创建者 ID 相同的用户进程,以及超级用户进程可以执行这一操作。
IPC_SET:按参数 buf 指向的结构中的值设置该消息队列对应的 msqid_ds 结构。只有有效用户 ID 和消息队列的所有者 ID 或创建者 ID 相同的用户进程,以及超级用户进程可以执行这一操作。
IPC_STAT:获得该消息队列的 msqid_ds 结构,保存于 buf 指向的缓冲区。

应用消息队列的 demo

下面是一个通过消息队列进行进程间通信的 demo:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h> int main(void)
{
int msgid;
int status;
char str1[ ]={"test message:hello!"};
char str2[ ]={"test message:godbye!"};
struct msgbuf
{
long msgtype;
char msgtext[];
}sndmsg, rcvmsg; if((msgid=msgget(IPC_PRIVATE,))==-)
{
printf("msgget error!\n");
exit();
}
sndmsg.msgtype = ;
sprintf(sndmsg.msgtext,"%s", str1);
if(msgsnd(msgid,(struct msgbuf *)&sndmsg,sizeof(str1)+,)==-)
{
printf("msgsnd error!\n");
exit();
}
sndmsg.msgtype = ;
sprintf(sndmsg.msgtext, "%s", str2);
if(msgsnd(msgid,(struct msgbuf *)&sndmsg,sizeof(str2)+,)==-)
{
printf("msgsnd error!\n");
exit();
}
if((status=msgrcv(msgid,(struct msgbuf *)&rcvmsg,,,IPC_NOWAIT))==-)
{
printf("msg rcv error!\n");
exit();
} printf("The received message: %s.\n", rcvmsg.msgtext);
// 下面的代码会删除消息队列,这里把它注释掉是为了使用 ipcs 命令进行观察
// msgctl(msgid, IPC_RMID,0);
exit();
}

简单起见,该程序自己完成了消息的发送和接收。由于我们指定了接收消息的类型,所以只有第二条消息会被接收。
把程序代码保存到文件 msgqueue.c 中,并编译:

$ gcc -Wall msgqueue.c -o msgqueue_demo

然后运行程序:

$ sudo ./msgqueue_demo

接收者只收到了类型为 222 的消息。
由于我们注释了程序中删除消息队列的代码,所以我们还可以通过 ipcs 命令来查看程序中创建的消息队列:

$ ipcs -q

总结

本文以一个极简的 demo 介绍并演示了 IPC 消息队列的基本概念和用法,对于了解 IPC 消息队列我想这些已经足够了。

参考:
《Linux 环境下 C 编程指南》

System V IPC 之消息队列的更多相关文章

  1. System V IPC(1)-消息队列

    一.概述                                                    System V三种IPC:消息队列,信号量,共享内存.这三种IPC最先出现在AT&am ...

  2. 四十九、进程间通信——System V IPC 之消息队列

    49.1 System V IPC 介绍 49.1.1 System V IPC 概述 UNIX 系统存在信号.管道和命名管道等基本进程间通讯机制 System V 引入了三种高级进程间通信机制 消息 ...

  3. Linux 系统编程 学习:04-进程间通信2:System V IPC(1)

    Linux 系统编程 学习:04-进程间通信2:System V IPC(1) 背景 上一讲 进程间通信:Unix IPC-信号中,我们介绍了Unix IPC中有关信号的概念,以及如何使用. IPC的 ...

  4. Linux 系统编程 学习:05-进程间通信2:System V IPC(2)

    Linux 系统编程 学习:05-进程间通信2:System V IPC(2) 背景 上一讲 进程间通信:System V IPC(1)中,我们介绍了System IPC中有关消息队列.共享内存的概念 ...

  5. 第3章 System V IPC

    3.1 概述 System V IPC 包含:System V消息队列.System V信号量.System V共享内存. 3.2 key_t 键和 ftok函数 这三种类型的System V IPC ...

  6. 《Unix网络编程》卷2 读书笔记 第3章- System V IPC

    1. 概述 三种类型的System V IPC:System V 消息队列.System V 信号量.System V 共享内存区 System V IPC在访问它们的函数和内核为它们维护的信息上共享 ...

  7. 从并发处理谈PHP进程间通信(二)System V IPC

    .container { margin-right: auto; margin-left: auto; padding-left: 15px; padding-right: 15px } .conta ...

  8. System V IPC 之共享内存

    IPC 是进程间通信(Interprocess Communication)的缩写,通常指允许用户态进程执行系列操作的一组机制: 通过信号量与其他进程进行同步 向其他进程发送消息或者从其他进程接收消息 ...

  9. System V IPC

    1.概述 System V IPC共有三种类型:System V消息队列.System V 信号量.System V 共享内存区. System V IPC操作函数如下: 2.key_t键和ftok函 ...

随机推荐

  1. 小结:c++中的new、operator new和placement new

    小结:c++中的new.operator new和placement new new(也称作new operator),是new 操作符,不可重载 class T{...}; T *t = new T ...

  2. PHP之工厂方法模式(三)

    定义 ​ 定义一个用于创建对象的接口(抽象工厂类),让子类决定实例化哪一个类,工厂方法使得一个类的实例化延迟到其子类(抽象工厂类的子类). ​ 工厂方法模式是简单工厂模式的进一步抽象和推广.在简单工厂 ...

  3. JavaWeb开发环境搭建Eclipse配置Tomcat

    转载请标明出处:http://blog.csdn.net/wu_wxc/article/details/48651251本文出自[吴孝城的CSDN博客] 工具: Eclipse官网下载:http:// ...

  4. 如何使用Git以及GitHub

    Git在程序的版本控制上有着极大的优势,下面是简单对其的简介 Git 的特点: 1 Snapshots, Not Differences 直接记录快照而非差异对比. 传统的版本控制系统(version ...

  5. 转: web 页面加载速度优化实战-100% 的飞跃提升

    前言 一个网站的加载速度有多重要? 反正我相信之前来 博主网站 的人至少有 50% 在加载完成前关闭了本站. 为啥捏? 看图 首页完整加载时间 8.18s,看来能进来看博主网站的人都是真爱呀,哈哈. ...

  6. 利用XAMPP本地搭建WordPress博客

    现在越来越多的人利用WordPress搭建了自己的博客网站,我也是一样,但是还有一些人不知道怎么搭建WordPress网站的方法,因为怕弄 不好,所以也就没有花钱去做,所以这里我就讲讲怎么样利用XAM ...

  7. C# 发送手机短信

    由于使用 收费 第三方 屏蔽   可能有 免费 第三方. private const string Cdkey = "8S*********************";密钥id p ...

  8. 提高UI设计效率的4个技巧

    提高UI设计效率的4个技巧 如何提高UI设计的速度?在这里分享一些我观察到的常见问题和改善方式.当然,需要注意的地方何其多. 身为设计师的你,应该要了解工程实作的基本原理: 业界 NG 率:接近 10 ...

  9. Ubuntu 14.04下Hadoop2.4.1集群安装配置教程

    一.环境 系统: Ubuntu 14.04 64bit Hadoop版本: hadoop 2.4.1 (stable) JDK版本: OpenJDK 7 台作为Master,另3台作为Slave. 所 ...

  10. POJ-1004-Finanical Management

    Description Larry graduated this year and finally has a job. He's making a lot of money, but somehow ...