1、创建消息队列 — msgget()
#include<stdio.h>
int msgget(key_t key,int msgflg);
功能:
创建一个新的或打开一个已经存在的消息队列。不同的进程调用此函数,只要用相同的key值就能得到同一个消息队列的标识符。
参数:
key: IPC键值,唯一的键值确定唯一的消息队列。
方法1:任意指定一个数。
方法2:使用ftok函数获取键值。
msgflg: 标识函数的行为及消息队列的权限。
msgflg的取值:
取值 | 作用 |
---|---|
IPC_CREAT | 创建消息队列 |
IPC_EXCL | 检测消息队列是否存在 |
位或权限位 | 消息队列位或权限位后可以设置消息队列的访问权限,格式和open函数的mode_t一样,但可执行权限未使用 |
返回值:
成功:消息队列的标识符
失败:返回 -1
使用shell命令操作消息队列 :
ipcs -q 查看消息队列
ipcrm -q msqid 删除消息队列
案例代码:
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/msg.h>
int main(int argc,char const *argv[])
{
key_t mykey;
if((mykey = ftok(".",100)) == -1)
{
perror("fail to ftok\n");
exit(1);
}
printf("mykey = %#x\n",mykey);
int msqid;
if((msqid = msgget(mykey,IPC_CREAT|0666)) == -1)
{
perror("fail to msgget\n");
exit(1);
}
printf("msqid = %d\n",msqid);
system("ipcs -q");
return 0;
}
执行结果:
2、发送消息 – msgsnd()
#include<sys/msg.h>
int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);
功能:
将新消息添加到消息队列。
参数:
msqid:消息队列的标识符(id)。
msgp:待发送消息结构体的地址。
msgsz:消息正文的字节数,不包括消息的编号长度。
msgflg:函数的控制属性(标志位 )。
msgflg的取值:
取值 | 作用 |
---|---|
0 | msgsnd调用阻塞直到条件满足为止 (阻塞) |
IPC_NOWAIT | 若消息没有立即发送则调用该函数的进程会立即返回 (非阻塞) |
返回值:
成功:返回 0
失败:返回 -1
案例代码:
#include<stdio.h>
#include<stdlib.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/msg.h>
#include<sys/ipc.h>
#define N 128
typedef struct
{
long msg_type; // 消息类型,必须在结构体的第一个位置并且类型必须是long
char msg_text[N]; // 消息正文,也可以有多个成员并且类型也可以是任意
}MSG;
#define MSGTEXT_SIZE (sizeof(MSG) - sizeof(long))
int main(int argc,char const *argv[])
{
//使用ftok函数获取键值
key_t key;
if((key = ftok(".",100)) == -1)
{
perror("faile to ftok");
exit(1);
}
//使用msgget函数创建一个消息队列
int msgid;
if((msgid = msgget(key,IPC_CREAT|0777)) == -1)
{
perror("fail to msgget");
exit(1);
}
system("ipcs -q");
//使用msgsnd函数向消息队列中发送数据(写操作)
MSG msg1 = {1,"hello world"};
MSG msg2 = {4,"nihao beijing"};
MSG msg3 = {2,"hello kitty"};
MSG msg4 = {3,"welcome to nangong"};
if(msgsnd(msgid,&msg1,MSGTEXT_SIZE,0) == -1)
{
perror("fail to msgsnd");
exit(1);
}
if(msgsnd(msgid,&msg2,MSGTEXT_SIZE,0) == -1)
{
perror("fail to msgsnd");
exit(1);
}
if(msgsnd(msgid,&msg3,MSGTEXT_SIZE,0) == -1)
{
perror("fail to msgsnd");
exit(1);
}
if(msgsnd(msgid,&msg4,MSGTEXT_SIZE,0) == -1)
{
perror("fail to msgsnd");
exit(1);
}
system("ipcs -q");
return 0;
}
执行结果:
3、接收消息 --msgrcv()
#include<sys/msg.h>
ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp, int msgflg);
功能:
从标识符为msqid的消息队列中接受一个消息。一旦接受消息成功,则消息在消息队列中被删除。
参数:
msqid: 消息队列的标识符,代表要从哪个消息列中获取消息。
msgp: 存放消息结构体的地址。
msgsz: 消息正文的字节数。
msgtyp: 消息的类型、可以有以下几种类型
类型 | 作用 |
---|---|
msgtyp = 0 | 返回队列中的第一个消息 |
msgtyp > 0 | 返回队列中消息类型为msgtyp的消息 |
msgtyp < 0 | 返回队列中消息类型值小于或等于msgtyp绝对值的消息,如果这种消息有若干个,则取类型值最小的消息。 |
msgflg: 标志位
0 阻塞
IPC_NOWAIT 非阻塞
案例代码:
#include<stdio.h>
#include<unistd.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/msg.h>
#include<stdlib.h>
#define N 128
typedef struct
{
long msg_type;
char msg_text[N];
}MSG;
#define MSGTEXT_SIZE (sizeof(MSG) - sizeof(long))
int main()
{
key_t key;
if((key = ftok(".",100)) == -1)
{
perror("fail to ftok");
exit(1);
}
//使用msgget函数创建一个消息队列
int msgid;
if((msgid = msgget(key,IPC_CREAT|0777)) == -1)
{
perror("fail to msgget");
exit(1);
}
system("ipcs -q");
//通过msgrcv函数接收消息队列中的信息(读操作)
MSG msg;
if(msgrcv(msgid,&msg,MSGTEXT_SIZE,0,0) == -1)
{
perror("fail to msgrcv");
exit(1);
}
printf("recv_msg = %s\n",msg.msg_text);
system("ipcs -q");
return 0;
}
执行结果:
4、消息队列的控制 --msgctl()
#include<sys/msg.h>
int msgctl(int msqid, int cmd, stuct msqid_ds *buf);
功能:
对消息队列进行各种控制,如修改消息队列的属性,或删除消息队列。
参数:
msqid:消息队列的标识符。
cmd:函数功能的控制。
cmd的取值
取值 | 作用 |
---|---|
IPC_RMID | 删除由msqid指示的消息队列,将它从系统中删除并破坏相关数据结构 |
IPC_STAT | 讲msqid相关的数据结构中各个元素的当前值存入到由buf指向的结构中 |
IPC_SET | 将msqid相关的数据结构中的元素设置为由buf指向的结构中的对应值 |
buf:msqid_ds数据类型的地址,用来存放或更改消息队列的属性。
返回值:
成功:返回 0
失败:返回 -1