目录
一.特点
消息队列是IPC对象(活动在内核级别的一种进程间通信的工具)的一种
一个消息队列由一个标识符 (即队列ID)来标识
消息队列就是一个消息的列表。用户可以在消息队列中添加消息、读取消息等
消息队列可以按照类型(自己设一个值作为类型)来发送/接收消息
二.步骤
(1) 创建key值: ftok
(2) 创建或打开消息队列: msgget
(3) 添加消息:按照消息的类型把消息添加到已经打开的消息队列末尾 msgsnd
(4) 读取消息:可以按照消息类型把消息从消息队列中取走 msgrcv
(5) 删除消息队列: msgctl
三. 操作命令
ipcs -q: 查看消息队列
ipcrm -q msgid: 删除消息队列
四.函数接口
1.创建或打开消息队列
#include <sys/msg.h>
int msgget(key_t key, int flag);
功能:创建或打开一个消息队列
参数: key值
flag:创建消息队列的权限IPC_CREAT|IPC_EXCL|0666
返回值:成功:msgid
失败:-1
2.添加消息
int msgsnd(int msqid, const void *msgp, size_t size, int flag);
功能:添加消息
参数:msqid:消息队列的ID
msgp:指向消息的指针。常用消息结构msgbuf如下:
struct msgbuf{
long mtype; //消息类型
char mtext[N]}; //消息正文
size:发送的消息正文的字节数
flag:IPC_NOWAIT消息没有发送完成函数也会立即返回
0:直到发送完成函数才返回
返回值:成功:0
失败:-1
用法: msgsnd(msgid, &msg, sizeof(msg)-sizeof(long),0);
3.读取消息
int msgrcv(int msgid, void* msgp, size_t size, long msgtype, int flag);
功能:读取消息
参数:msgid:消息队列的ID
msgp:存放读取消息的空间
size:接受的消息正文的字节数(sizeof(msgp)-sizeof(long))
msgtype:
0:接收消息队列中第一个消息。
大于0:接收消息队列中第一个类型为msgtyp的消息.
小于0:接收消息队列中类型值不小于msgtyp的绝对值且类型值又最小的消息。
flag:
0:若无消息函数会一直阻塞
IPC_NOWAIT:若没有消息,进程会立即返回ENOMSG
返回值:成功:接收到的消息的长度
失败:-1
4.删除
int msgctl ( int msgqid, int cmd, struct msqid_ds *buf );
功能:对消息队列的操作,删除消息队列
参数:msqid:消息队列的队列ID
cmd:
IPC_STAT:读取消息队列的属性,并将其保存在buf指向的缓冲区中。
IPC_SET:设置消息队列的属性。这个值取自buf参数。
IPC_RMID:从系统中删除消息队列。
buf:消息队列缓冲区
返回值:成功:0
失败:-1
用法:msgctl(msgid, IPC_RMID, NULL);
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <errno.h>
struct msgbuf
{
long type; //必须有,在第一个,表示消息的类型,值>0!
int num; //消息正文,自己定义
char ch;
};
int main(int argc, char const *argv[])
{
key_t key;
int msgid;
if ((key = ftok("msg.c", 'a')) < 0)
{
perror("ftok err");
return -1;
}
printf("key: %#x\n", key);
//打开或创建消息队列
msgid = msgget(key, IPC_CREAT | IPC_EXCL | 0777);
if (msgid <= 0)
{
if (errno == EEXIST)
msgid = msgget(key, 0777); //如果已经存在消息队列那直接打开该消息队列
else
{
perror("msgget err");
return -1;
}
}
printf("msgid: %d\n", msgid);
//添加消息
struct msgbuf msg;
msg.type = 10;
msg.num = 1000;
msg.ch = 'a';
msgsnd(msgid, &msg, sizeof(msg) - sizeof(long), 0); //0:发完消息再返回,而不是立即返回函数
msg.type = 20;
msg.num = 2000;
msg.ch = 'b';
msgsnd(msgid, &msg, sizeof(msg) - sizeof(long), 0);
//读取消息
struct msgbuf m;
msgrcv(msgid, &m, sizeof(m) - sizeof(long), 20, 0); //0:阻塞,读完消息再返回
printf("%d %c\n", m.num, m.ch);
//删除消息队列
msgctl(msgid, IPC_RMID, NULL);
return 0;
}
进程间通信