上文链接:进程间通信(一)
续进程间通信
三、消息队列
消息队列是消息的链接表,存放在内核中,一个消息队列由一个标识符(即队列ID)来标识。
1、特点
1)消息队列是面向记录的,其中的消息具有特定的格式以及特定的优先级;
2)消息队列独立于发送与接收进程。进程终止时,消息队列及其内容并不会被删除;
3)消息队列可以实现消息的随机查询,消息不一定要以先进先出的次序读取,也可以按消息的类型读取。
2、原型
#include <sys/msg.h>
//创建或打开消息队列;成功返回队列D,失败返回0
int msgget(key_t key,int flag);
//添加消息:成功返回0,失败返回-1
int msgsnd(int msqid,const void *ptr,size_t size,int flag);
//读取消息:成功返回消息数据的长度,失败返回-1
int msgrcv(int msqid,void *ptr,size_t size,long type,int flag);
//控制消息队列:成功返回0,失败返回-1
int msgctl(int msqid,int cmd,struct msqid_ds *buf);
msgget创建一个新的消息队列
1、如果没有与键值key相对应的消息队列,并且flag中包含了IPC_CREAT标志位。
2、key参数为IPC_PRIVATE。
msgrcv在读取消息队列时,type参数有下面几种情况:
1、type == 0,返回队列中的第一个消息;
2、type > 0,返回队列中消息类型为 type 的第一个消息;
3、type < 0,返回队列中消息类型值小于或等于 type 绝对值的消息,如果有多个,则取类型值最小的消息。
以下为消息队列编程收发数据
接收
#include<stdio.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/msg.h>
#include<string.h>
struct msgbuf{
long mtype;
char mtext[32];
};
int main(){
struct msgbuf readBuf;
int msgId=msgget(0x3250,IPC_CREAT|0777);
if(msgId==-1){
printf("msgget failed!\n");
}
msgrcv(msgId,&readBuf,sizeof(readBuf.mtext),321,0);
printf("read msg:%s \n",readBuf.mtext);
return 0;
}
发送
#include<stdio.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/msg.h>
#include<string.h>
struct msgbuf{
long mtype;
char mtext[32];
};
int main(){
struct msgbuf sendBuf={321,"hello friend!"};
int msgId=msgget(0x3250,IPC_CREAT|0777);
if(msgId==-1){
printf("msgget failed!\n");
}
msgsnd(msgId,&sendBuf,strlen(sendBuf.mtext),0);
return 0;
}
ftok函数
系统建立IPC通讯(如消息队列、共享内存时)必须指定一个ID值。通常情况下,该id值通过ftok函数得到
。
ftok原型如下:
#include <sys/types.h>
#include <sys/ipc.h>
key_t ftok( char * fname, int id )
fname:即你指定的文件名(该文件必须是存在而且可以访问的),或路径名,
id:是子序号,一个整数 。虽然为int,但是只有8个比特被使用(0-255)。
返回值:成功执行key_t值将会被返回;失败返回-1。
可以这样用:
key_t key;
key=ftok(".",1); //当前路径下,子序列号为1
在一般的UNIX实现中,是将文件的索引节点号取出,前面加上子序号得到key_t的返回值。如指定文件的索引节点号为65538,换算成16进制为 0x010002,而你指定的ID值为38,换算成16进制为0x26,则最后的key_t返回值为0x26010002。
查询文件索引节点号的方法是: ls -ai
server.c
#include<stdio.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/msg.h>
#include<string.h>
struct msgbuf{
long mtype;
char mtext[32];
};
int main(){
struct msgbuf readBuf;
key_t key;
key=ftok(".",1);
printf("key:%x\n",key);
int msgId=msgget(key,IPC_CREAT|0777);
if(msgId==-1){
printf("msgget failed!\n");
}
msgrcv(msgId,&readBuf,sizeof(readBuf.mtext),321,0);
printf("read msg:%s \n",readBuf.mtext);
struct msgbuf sendBuf={256,"welcome here!"};
msgsnd(msgId,&sendBuf,strlen(sendBuf.mtext),0);
msgctl(msgId,IPC_RMID,NULL);
return 0;
}
client.c
#include<stdio.h>
#include<sys/types.h>
#include<sys/ipc.h>
#include<sys/msg.h>
#include<string.h>
struct msgbuf{
long mtype;
char mtext[32];
};
int main(){
struct msgbuf sendBuf={321,"hello friend!"};
struct msgbuf readBuf;
key_t key;
key=ftok(".",1);
printf("key:%x\n",key);
int msgId=msgget(key,IPC_CREAT|0777);
if(msgId==-1){
printf("msgget failed!\n");
}
msgsnd(msgId,&sendBuf,strlen(sendBuf.mtext),0);
printf("send over\n");
msgrcv(msgId,&readBuf,sizeof(readBuf.mtext),256,0);
printf("read from get:%s\n",readBuf.mtext);
msgctl(msgId,IPC_RMID,NULL);
return 0;
}