进程间通信(二)

上文链接:进程间通信(一)
续进程间通信

三、消息队列

消息队列是消息的链接表,存放在内核中,一个消息队列由一个标识符(即队列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;
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值