您的位置:首页 > 运维架构 > Linux

Linux 消息队列编程

2016-03-11 09:30 337 查看
消息队列、信号量以及共享内存被称作 XSI IPC,它们均来自system V的IPC功能,因此具有许多共性。
键和标识符:

  内核中的每一种IPC结构(比如信号量、消息队列、共享内存)都用一个非负整数的标识符加以标示(如共享内存的shmid、信号量的semid、以及消息队列的msgid)。不同于文件描述符,IPC标识符不是一个小的非负整数,它是一个int型的整数,当一个标识符被创建,以后又被删除时,这个整数持续加1,达到整型的最大值后,重新回到0。

  但是每一个IPC对象在内核中的标识符只能在内部被识别,为了让不同的进程能够在同一个IPC对象上汇合,还需要一个外部的标识来表示一个IPC 对象,这就是key 键值。或者可以这样理解:标识符是一个打开了的IPC对象的描述符,而键值则让进程获得这个标识符。

  当我们通过一个键值创建了一个IPC对象以后,就可以用这个IPC对象的标识符操作这个IPC对象。例如当进程用指定的键值获得一个共享内存的标识符shmid以后,就可以通过这个标识符将共享内存映射到自身的地址空间上,当然后续对共享内存的操作也是基于这个标识符。

消息队列编程:

  消息队列作为XSI IPC的一种,许多实现和操作和前面描述的信号量和共享内存有很多相似的地方。先简述一下消息队列的原理:消息队列的本质就是一个消息的链表,而每个消息的结构如下:

struct msgbuf {
  ong mtype; /* message type, must be > 0 */
char mtext[1]; /* message data */
};


mtype:消息的类型        mtext[]: 消息中的数据

也就是说消息队列就是一个数据域是以上结构体的链表。既然了解了一个消息队列的本质就是一个链表,可以想象,如果进程间要通过消息队列通信,那么发送消息的进程的主要工作就是要构建一个数据域,然后交给内核来插入到链表中,而接收消息的进程就是通过内核函数来将数据从消息队列中取出来。消息队列还有如下特点:

 *消息队列的消息类型可以不同,进程取出消息时可以指定消息类型取出需要的消息

 *当进程取出一条消息后,该消息会立即被移出消息队列

利用消息队列通信主要由以下操作完成:

 *创建/打开消息队列   :int msgget(key_t key, int msgflg);

 *发送消息                 :int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg)

 *接收消息                 :ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp,int msgflg)

 *删除消息队列           :int msgctl(int msqid, int cmd, struct msqid_ds *buf)

这些函数都可以通过man命令查到具体的用法,这里就不在详细解释,下面是一个测试的范例:

send 程序:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdio.h>

#define TEXT_SZ 2048
struct msgt
{
long msgtype;
char msgtext[TEXT_SZ];
};

int main()
{
int msgid;
key_t key;
int running = 1;
struct msgt msg_data;
int msgtype;
key = ftok("/home/application/massage_queue",2);
//创建消息队列
msgid  = msgget(key, IPC_CREAT);
//循环
while(running)
{
printf("Please Input msgtype,Input 0 to quit!\n");
scanf("%d",&msgtype);
printf("Please Input datas!\n");

//从终端读取数据
scanf("%s",msg_data.msgtext);
//将数据写入消息队列
msg_data.msgtype = msgtype;
msgsnd(msgid, &msg_data, sizeof(msg_data), 0);

if(strncmp(msg_data.msgtext,"end",3)==0)
{
msgsnd(msgid, "end", 3, 0);
running = 0;
}
}
//删除消息队列
msgctl(msgid, IPC_RMID, 0);
return 0;

}


receive 程序:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdio.h>
#include <unistd.h>

#define TEXT_SZ 2048
struct msgt
{
long msgtype;
char msgtext[TEXT_SZ];
};

int msgid;
void childprocess()
{
struct msgt msg_d;
int running = 1;
while(running)
{
//接受消息队列中的数据
msgrcv(msgid, &msg_d, sizeof(msg_d), 0,0);
//打印数据
printf("Receive datas from queue:%s",msg_d.msgtext);
if(strncmp(msg_d.msgtext,"end",3)==0)
{
running = 0;
}
}

}
int main()
{

key_t key;
pid_t pid;
int i;
key = ftok("/home/application/massage_queue",2);
//打开消息队列
msgid  = msgget(key, IPC_EXCL);

for(i=0;i<3;i++)
{
pid = fork();
if(pid<0)
{
printf("fork error!\n");
}
else if(pid==0)
{
childprocess();
}
}

return 0;
}


当两个程序运行起来以后可以发现通过send成序发送的数据在receive 中可以接收到,说明这些函数的调用很成功,同时也证明了消息队列的通信是成功的。

 
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: