linux消息队列通信

程序目的:学习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
……
***********************************************************/
本人初学者,在写本博文中可能有一些不当的地方,欢迎大家向我指正。

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值