一、消息队列的定义    

    消息队列能够弥补管道的不足,实现双向交互数据,是一个进程向另一进程发送进程块的方法。与管道不同的是,管道是基于字节流的,消息队列是基于消息的,且消息队列的读取不一定是先进先出。

二、消息队列的创建

    通过函数int messget(key_t key,int msgflg);创建

   key:端口号,可以有 ftok生成。

  msgflg:

    IPC_CRTAT 若果 IPC不存在,则创建一个IPC资源,

    IPC_EXCL:一般和 IPC_CREAT一起使用可以保证所得的对象是新建的,而不是打开已有的。

三、向消息队列读数据

    msgrcv从队列中取的消息

       ssize_t msgrcv(int msqid,void *msgp,size_t msgsz,long msgtyp,int msgflg);

    msgsnd将消息放进队列中

       int msgsnd(int msqid,const void *msgp,size_t msgsz,int msgflg);

四、设置消息队列属性

    int msgctl(int msgqid,int cmd,struct msqid_ds *buf);

*** 可以用man 函数名 查看函数***

程序代码:

coom.h

  1 #pragma once
  2 #include
  3 #include
  4 #include
  5 #include
  6 #include
  7 #include
  
  8 #define _PATH_ "."
  9 #define _PROCID_ 0x776
 10 #define _SIZE 1024
 11 #define SERVE_MSGTYPE 1
 12 #define CLIECT_MSGTYPE 2
 
 13 typedef struct message
 14 {
 15   long mtype;
 16   char mtext[_SIZE];
 17 }message;
 18 int create_msg(int msg_num);
 19 int drstroy_msg(int msg_id);
 20 int set_msg();
 21 int get_msg();
 22 int msg_send(int msg_id,const char* msg,long type);
 23 int msg_rec(int msg_id ,char buf[],long type);

comm .c

  1 #include"comm.h"
  2  int create_msg(int msg_num)
  3 {
  4   key_t _key=ftok(_PATH_,_PROCID_);
  5   if(_key < 0)
  6   {
  7     perror("ftok");
  8     return -1;
  9   }
 10   int ret= msgget(_key,msg_num);
 11    if(ret < 0)
 12    {
 13     perror("msgget");
 14     return -1;
 15    }
 16  return ret;
 17 }
 18 int drstroy_msg(int msg_id)
 19 {
 20   if(msgctl(msg_id,IPC_RMID,NULL) < 0)
 21   {
 22     perror("msgctl");
 23     return -1;
 24   }else{
 25     printf("remove message_queue\n");
 26     return 0;
 27   }
 28 }
 29 int set_msg()
 30 {
 31   umask(0);
 32   return create_msg(IPC_CREAT |IPC_EXCL |0666);
 33 }
 34 int get_msg()
 35 {
 36     return create_msg(IPC_CREAT);
 37 }
 38 int msg_send(int msg_id,const char* msg,long type)
 39 {
 40   struct message _msg;
 41   _msg.mtype=type;
 42   strcpy(_msg.mtext,msg);
 43   if(msgsnd(msg_id,&_msg,strlen(_msg.mtext),0)< 0)
 44   {
 45     perror("msg_send error");
 46     return -1;
 47   }
 48   return 0;
 49 }
 50 int msg_rec(int msg_id,char buf[],long type)
 51 {
 52   struct message _msg;
 53   memset(_msg.mtext,'\0',_SIZE);
 54   if(msgrcv(msg_id,&_msg,_SIZE,type,0)< 0)
 55   {
 56     perror("msg_receve error");
 57     return -1;
 58   }
 59   strcpy(buf,_msg.mtext);
 60   return 0;
 61 }
 62

client.c

  1 #include"comm.h"
  2 int main()
  3 {
  4  int msg_id=get_msg();
  5  if(msg_id<0)
  6  {
  7   perror("error");
  8   exit(1);
  9  }
 10   char buf[_SIZE];
 11  while(1)
 12  {
 13   fflush(stdout);
 14   printf("please client input: ");
 15   memset(buf,'\0',_SIZE);
 16   fgets (buf,sizeof(buf),stdin);
 17   if(msg_send(msg_id,buf,CLIECT_MSGTYPE)< 0)
 18     {
 19       perror("send fail");
 20       exit(1);
 21     }
 22 
 23   if(msg_rec(msg_id,buf,SERVE_MSGTYPE)<0)
 24     {
 25       perror("recve fail");
 26       exit(1);
 27     }
 28   printf("serve:%s" ,buf);
 29  }
 30   return 0;
 31 }

server.c

  1 #include "comm.h"
  2 int main()
  3 {
  4   int msg_id=set_msg();
  5   if(msg_id<0)
  6   {
  7     perror("mig_id");
  8     exit(1);
  9   }
 10   printf("%d\n",msg_id);
 11   char buf[_SIZE];
 12   while(1)
 13   {
 14      if(msg_rec(msg_id,buf,CLIECT_MSGTYPE)< 0)
 15     {
 16       perror("recve fail");
 17       exit(1);
 18     }
 19     else
 20     {
 21       printf("client :%s",buf);
 22     }
 23     printf("server please input: ");
 24     fflush(stdout);
 25     memset(buf,'\0',_SIZE);
 26     fgets(buf,_SIZE,stdin);
 27     if(msg_send(msg_id,buf,SERVE_MSGTYPE)<0)
 28     {
 29       perror("send fail");
 30       exit(1);
 31     }
 32   }
 33   drstroy_msg(msg_id);
 34   return 0;
 35 }

运行结果:

Linux --进程间通信--消息队列_第1张图片