linux.调整收发队列,linux消息队列通信

程序目的:学习linux消息队列通信

所用主要函数:msgget(),msgsnd(),msgrcv(),msgctl()

首先介绍每个函数的用法:

(1)msgget

使用格式:

#include

#include

#include

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

#include

#include

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

#include

#include

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

#include

#include

#include

#include

#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

#include

#include

#include

#include

#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
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值