1.特点:
  消息队列是IPC对象的一种
  消息队列由消息队列ID来唯一标识
  消息队列就是一个消息的列表。用户可以在消息队列中添加消息、读取消息等。
  消息队列可以按照类型来发送/接收消息(消息的类型是正整数)

2.步骤:
  1)产生key值ftok
  2)创建或打开消息队列
  3)添加消息:按照类型把消息添加到已打开的消息队列末尾
  4)读取消息:可以按照类型把消息从消息队列中取走
  5)删除消息队列

3.相关函数:
  1)int msgget(key_t key, int flag);
    功能:创建或打开一个消息队列
    参数: key值
          flag:创建消息队列的权限IPC_CREAT|IPC_EXCL|0666
    返回值:成功:msgid ; 失败:-1

  2)int msgsnd(int msqid, const void *msgp, size_t size, int flag);
    功能:添加消息
    参数:msqid:消息队列的ID
         msgp:指向消息的指针。常用消息结构msgbuf如下:
          struct msgbuf{
              long mtype;  //消息类型100 100
              char mtext[N] ; //消息正文

          };
       size:发送的消息正文的字节数
         flag:IPC_NOWAIT消息没有发送完成函数也会立即返回 ; 0:直到发送完成函数才返回
    返回值:成功:0 ; 失败:-1
  使用:msgsnd(msgid, &msg,sizeof(msg)-sizeof(long), 0)
  注意:消息结构除了第一个成员必须为long类型外,其他成员可以根据应用的需求自行定义。

  3)int msgrcv(int msgid,  void* msgp,   size_t size,  long msgtype,  int flag);
    功能:读取消息
    参数:msgid:消息队列的ID
         msgp:存放读取消息的空间
       size:接受的消息正文的字节数
       msgtype:
          0:接收消息队列中第一个消息。
          大于0:接收消息队列中第一个类型为msgtyp的消息.
          小于0:接收消息队列中类型值不小于msgtyp的绝对值且类型值又最小的消息。
       flag:0:若无消息函数会一直阻塞 ; IPC_NOWAIT:若没有消息,进程会立即返回ENOMSG
    返回值:成功:接收到的消息的长度 ; 失败:-1

  4)int msgctl ( int msgqid, int cmd, struct msqid_ds *buf );
    功能:对消息队列的操作,删除消息队列
    参数:msqid:消息队列的队列ID
       cmd:
        IPC_STAT:读取消息队列的属性,并将其保存在buf指向的缓冲区中。
        IPC_SET:设置消息队列的属性。这个值取自buf参数。
        IPC_RMID:从系统中删除消息队列。
       buf:消息队列缓冲区
    返回值:成功:0 ; 失败:-1
  用法:msgctl(msgid, IPC_RMID, NULL)

【3】比较:
无名管道 pipe: 具有亲缘关系的进程间,单工,数据在内存中

有名管道 fifo: 可用于任意进程间,双工,有文件名,数据在内存

信号 signal: 唯一的异步通信方式

消息队列 msg:常用于cs模式中, 按消息类型访问 ,可有优先级

共享内存 shm:效率最高(直接访问内存) ,需要同步、互斥机制

信号灯集 sem:配合共享内存使用,用以实现同步和互斥

例子:同一进程下

#include <stdio.h>
#include <stdlib.h>
#include <errno.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h> struct msgbuf //消息结构体
{
long types;
int a;
char b;
};
struct msgbuf mymsgbuf, recvbuf; //定义消息结构变量 int main(int argc, const char *argv[])
{
key_t key;
int msgid;
mymsgbuf.types = ; //给消息结构赋值
mymsgbuf.a = ;
mymsgbuf.b = 'b'; key = ftok("./app",'a'); //获取key值
if(key < )
{
perror("ftok fail ");
exit();
}
// 创建消息队列,如果消息队列存在,errno 会提示 eexist
// 错误,此时只需要直接打开消息队列即可
msgid = msgget(key,IPC_CREAT|IPC_EXCL|);
if(msgid < )
{
if(errno == EEXIST) //文件存在错误提示
{
msgid = msgget(key,);//打开消息队列
}
else //其他错误退出
{
perror("msgget fail ");
exit();
}
}
//发送消息
msgsnd(msgid, &mymsgbuf, sizeof(mymsgbuf)-sizeof(long),); //接收消息
msgrcv(msgid,&recvbuf,sizeof(mymsgbuf)-sizeof(long),,); printf("mymsgbuf %d %c\n",mymsgbuf.a, mymsgbuf.b);
printf("recvbuf %d %c\n",recvbuf.a, recvbuf.b); //删除消息队列
msgctl(msgid, IPC_RMID, NULL); return ;
}

测试:

Linux 进程间通信 消息队列的更多相关文章

  1. Linux进程间通信-消息队列(mqueue)

    前面两篇文章分解介绍了匿名管道和命名管道方式的进程间通信,本文将介绍Linux消息队列(posix)的通信机制和特点. 1.消息队列 消息队列的实现分为两种,一种为System V的消息队列,一种是P ...

  2. 详解linux进程间通信-消息队列

    前言:前面讨论了信号.管道的进程间通信方式,接下来将讨论消息队列. 一.系统V IPC 三种系统V IPC:消息队列.信号量以及共享内存(共享存储器)之间有很多相似之处. 每个内核中的 I P C结构 ...

  3. Linux进程间通信—消息队列

    四.消息队列(Message Queue) 消息队列就是消息的一个链表,它允许一个或者多个进程向它写消息,一个或多个进程向它读消息.Linux维护了一个消息队列向量表:msgque,来表示系统中所有的 ...

  4. linux进程间通信-消息队列

    一 消息队列的介绍 消息队列提供了一种从一个进程向另一个进程发送一个数据块的方法. 每个数据块都被认为含有一个类型,接收进程可以独立地接收含有不同类型的数据结构. 我们可以通过发送消息来避免命名管道的 ...

  5. Linux 进程间通信 消息队列 实现两个进程间通信

    例子: 通过消息队列实现两个进程间通信,一个进程从终端输入数据,通过消息队列发送,另一个进程通过消息队列接收数据 文件1 创建进程1 终端输入通过消息队列发送数据 #include <stdio ...

  6. linux进程间通信消息队列:msgsnd: Invalid argument

    今天写了个消息队列的小测试程序结果send端程序总是出现:msgsnd: Invalid argument,搞了半个小时也没搞明白,后来查资料发现我将(st_msg_buf.msg_type = 0; ...

  7. PHP 进程间通信——消息队列(msg_queue)

    PHP 进程间通信--消息队列 本文不涉及PHP基础库安装.详细安装说明,请参考官网,或期待后续博客分享. 1.消息队列函数准备 <?php//生成一个消息队列的key$msg_key = ft ...

  8. Linux下进程间通信--消息队列

    消息队列的定义遍地都是,不想移驾,请看下文: 一.定义: 消息队列提供了一种从一个进程向另一个进程发送一个数据块的方法. 每个数据块都被认 为是有一个类型,接收者进程接收的数据块可以有不同的类型值.我 ...

  9. IPC进程间通信---消息队列

    消息队列 消息队列:消息队列是一个存放在内核中的消息链表,每个消息队列由消息队列标识符标识.与管道不同的是消息队 列存放在内核中,只有在内核重启(即操作系统重启)或者显式地删除一个消息队列时,该消息队 ...

随机推荐

  1. Sql Server 表结构相关

    1.库表列信息 --取所有库 SELECT Name FROM Master..SysDatabases ORDER BY Name --查询所有表 select name from 库名..syso ...

  2. 使用MySQL Workbench查询超时的错误

    MySQL Workbench是MySQL提供的连接工具,一直在用它.但是今天运行了一个SQL缺报出如下的错误: errcode 2013 lost connection to mysql serve ...

  3. HTTP/2的优先级

    前言 记得HTTP/3即将标准化了.今日早读文章由@smallbonelu翻译授权分享. @smallbonelu,一枚爱好跑步的前端工程师 正文从这开始-- 以正确的顺序请求页面资源对于快速的用户体 ...

  4. Gabor滤波器的理解

    搬以前写的博客[2014-02-28 20:03] 关于Gabor滤波器是如何提取出特征点,这个过程真是煎熬.看各种文章,结合百度.文章内部的分析才有一点点明白. Gabor滤波器究竟是什么?   很 ...

  5. python爬虫(2):图片,翻译爬虫

    import urllib.request#urllib.request.urlopen可以传入url或者Request对象#req=urllib.request.Request("http ...

  6. java程序中的多线程(转)

    为什么会排队等待? 下面的这个简单的 Java 程序完成四项不相关的任务.这样的程序有单个控制线程,控制在这四个任务之间线性地移动.此外,因为所需的资源 ― 打印机.磁盘.数据库和显示屏 -- 由于硬 ...

  7. Vue学习笔记【19】——Vue中的动画(使用第三方 CSS 动画库)

    导入动画类库:  <link rel="stylesheet" type="text/css" href="./lib/animate.css& ...

  8. EXCEL设置三级下拉框

    EXCEL设置三级下拉框 1.添加下拉框数据源 公式--->指定 公式--->名称管理器 2.设置第一级下拉框的值 3.第一级下拉框选出一个值 4.设置第二级下拉框(INDIRECT($A ...

  9. Android中当数据库需要更新时我们该怎么办?

    问题:Android数据库更新并保留原来的数据如何实现 Andoird的SQLiteOpenHelper类中有一个onUpgrade方法.帮助文档中只是说当数据库升级时该方法被触发.经过实践,解决了我 ...

  10. Linux 软硬链接区别

    一.“硬链接“和“软链接“ 链接的概念:链接简单说实际上是一种文件共享的方式,是 POSIX 中的概念,主流文件系统都支持链接文件. 链接的作用:可以将链接简单地理解为 Windows 中常见的快捷方 ...