程序目的:学习linux消息队列通信
所用主要函数:msgget(),msgsnd(),msgrcv(),msgctl()
首先介绍每个函数的用法:
(1)msgget
使用格式:
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgget(key_t k e y, int f l a g) ;
功能:是打开一个现存队列或创建一个新队列。
返回值:成功执行时,返回消息队列标识值。失败返回-1,有时也会返回0,这个时候也是可以正常使用的。
参数key:消息队列关联的键,可以直接赋值设为固定的的键值。也可以用ftok()。
也介绍一下ftok函数key_t ftok( char * fname, int id )
fname就是你指定的文件名(已经存在的文件名),一般使用当前目录;id是子序号。
这样就能得到一个固定的key_t值。
参数f l a g:消息队列的建立标志和存取权限,建立标志一般为指定为IPC_CREAT和IPC_EXCL标志 。
IPC_CREAT:如果内核中没有此队列,则创建它。当IPC_CREAT和IPC_EXCL一起使用时,如果队列已经存在,则失败。
例:msgget(ftok("./file",123),IPC_CREAT |0666)
(2)msgsnd
功能:在消息队列上进行收发消息。为了发送消息,调用进程对消息队列必须有写权能。接收消息时必须有读权能。
使用格式:
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);
返回值:成功执行时,返回消息队列标识值。失败返回-1。
参数:
msqid:消息队列的识别码。
msgp:指向消息缓冲区的指针,此位置用来暂时存储发送和接收的消息,是一个用户可定义的通用结构,如下
struct msgbuf {
long mtype; //消息类型,必须 大于0
char mtext[1]; // 消息文本
};
msgsz:消息的大小。
msgflg:用来指明核心程序在队列没有数据的情况下所应采取的行动。当msgflg为IPC_NOWAIT不会阻塞。当msgflg为0时msgsnd()在队列呈满或呈空的情形时,采取阻塞等待的处理模式。
(3)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);
参数:(同上2)
msgtyp:消息类型
其他参数同上(2)。
(4)msgctl
原型: int msgctl ( int msgqid, int cmd, struct msqid_ds *buf );
返回值: 如果成功为0 错误返回- 1。
参数msgqid :是消息队列对象的标识符。
第二个参数cmd:是函数要对消息队列进行的操作,它可以是:
IPC_STAT取出系统保存的消息队列的msqid_ds 数据,并将其存入参数buf 指向的msqid_ds 结构
中。
IPC_SET设定消息队列的msqid_ds 数据中的msg_perm 成员。设定的值由buf 指向的msqid_ds
结构给出。
IPC_EMID将队列从系统内核中删除。
这三个命令的功能都是明显的。唯一需要强调的是在IPC_STAT
命令中队列的msqid_ds 数据中唯一能被设定的只有msg_perm 成员,是ipc_perm 类型的
数据。而ipc_perm 中能被修改的只有mode,pid 和uid 成员。其他的都是只能由系统来设定
的。
******************************有关范例*********************************************
下面就看一个用消息队列通信写的一个简单的银行取号小程序:
在这里我用到了两个进程:msgc.c 和 msg.c
(1)msg.c
#include<stdio.h>
#include<stdlib.h>
#include<sys/ipc.h>
#include<sys/msg.h>
#include<string.h>
#define MSG_KEY 8888
struct mymesg{
long mtype;//消息类型
char mtext[512];//消息内容
};
int main(void)
{
int msgid;
struct msqid_ds buf;
struct mymesg mymsg;
msgid=msgget(MSG_KEY,IPC_CREAT |0600);//创建一个新队列
if(-1==msgid)
{
perror("msggit");
exit(EXIT_FAILURE);
}
while(1)
{
printf("请输入号码和内容:\n");
scanf("%d %s",&mymsg.mtype,mymsg.mtext);
msgsnd(msgid,&mymsg,strlen(mymsg.mtext)+1,0);//发送mymsg中的信息到msgid对应的消息队列
}
return 0;
}
(2)msg.c
#include<stdio.h>
#include<stdlib.h>
#include<sys/ipc.h>
#include<sys/msg.h>
#include<string.h>
#define MSG_KEY 8888
struct mymesg{
long mtype;
char mtext[512];
};
int main(void)
{
int msgid;
struct mymesg mymsgrcv;
msgid=msgget(MSG_KEY,IPC_CREAT |0600);
if(-1==msgid)
{
perror("msggit");
exit(EXIT_FAILURE);
}
while(1)
{
printf("请选择号码:");
scanf("%d",&mymsgrcv.mtype);
int ms=msgrcv(msgid,&mymsgrcv,512,mymsgrcv.mtype,0);//接收消息队列相应类型(号码)信息
if(-1==ms)
{
perror("msgrcv");
exit(EXIT_FAILURE);
}
printf("received:%s\n",mymsgrcv.mtext);
}
int msg=msgctl(msgid,IPC_RMID,0);//将队列从系统内核中删除。
if(-1==msg)
{
perror("msgctl");
exit(EXIT_FAILURE);
}
return 0;
}
/*********************程序运行结果*********************
[root@localhost msg]#gcc -o msg.c msg.c
[root@localhost msg]# ./msg.c第一个进程
请输入号码和内容:
1 Personal business
请输入号码和内容:
2 company business
请输入号码和内容:
3 Other business
[root@localhost msg]# gcc -o msgc.c msgc.c
[root@localhost msg]# ./msgc.c第二个进程
请选择号码:1
received:Personal business
请选择号码:2
received:company business
……
***********************************************************/
本人初学者,在写本博文中可能有一些不当的地方,欢迎大家向我指正。