什么是消息队列?

假设你是一个快递员,你需要将货物从一个仓库运到另一个仓库。但是你发现自己的时间不够用,需要另外请一个人来帮忙。那么,你们之间如何进行协作呢?

一种方式是直接将货物全部交给对方,但这样存在风险:对方可能会出现问题,导致货物丢失或损坏。

而另一种更安全的方式是,你将货物分批发送给对方,对方再按照你的要求逐批接收货物。这种方式类似于消息队列的通信方式。

在 Linux 系统中,消息队列是一种 IPC(进程间通信)机制,用于实现不同进程之间的通信。

简单地说,消息队列是一个消息的链表,消息发送方将消息发送到消息队列中,消息接收方从队列中读取消息。

消息队列的优点和缺点

与其他 IPC 机制相比,消息队列有以下优点:

  • 通过消息队列可以实现异步通信。
  • 消息队列可以存储多个消息,接收方可以按顺序逐个读取消息。
  • 消息队列的消息长度可以很长。

但是,消息队列也有以下缺点:

  • 消息队列的消息长度有限制,一般不能超过系统限制的最大值。
  • 消息队列需要调用特殊的系统调用来读写消息,开销较大。

消息队列的创建和使用方法

在Linux中,可以通过以下系统调用函数来创建和使用消息队列:

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h> 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); // 控制消息队列

其中,key是用来唯一标识消息队列的键值,msgflg是创建消息队列时的选项参数。在创建消息队列时,如果该键值已经存在,则直接返回该消息队列的标识符;如果不存在,则创建一个新的消息队列,并返回该消息队列的标识符。

在使用消息队列时,msgsnd函数用于向消息队列中发送消息,msgrcv函数用于从消息队列中接收消息,msgctl函数用于对消息队列进行控制,比如删除消息队列等。

消息队列的发送和接收示例

下面我们来看一个简单的示例,展示如何使用消息队列进行进程间通信。

假设有两个进程,一个发送进程和一个接收进程,它们之间需要传递一些数据。我们通过消息队列来实现进程间通信。

首先,我们需要创建一个消息队列,然后让发送进程向消息队列中发送一条消息,接收进程从消息队列中接收该消息,并进行处理。

创建消息队列

我们首先需要创建一个消息队列。可以使用msgget函数来创建消息队列。以下是创建消息队列的示例代码:

#include <sys/msg.h>
#include <stdio.h>
#include <stdlib.h> int main()
{
key_t key = ftok("/tmp", 'a'); // 创建一个唯一的key
int msgid = msgget(key, 0666 | IPC_CREAT); // 创建消息队列
if (msgid == -1) {
perror("msgget");
exit(EXIT_FAILURE);
} printf("消息队列创建成功,msgid=%d\n", msgid); return 0;
}

在上面的代码中,我们使用ftok函数创建一个唯一的key,这个key将作为消息队列的标识符。然后,我们使用msgget函数创建消息队列。如果创建成功,msgget函数将返回一个消息队列ID(msgid),否则将返回-1。在本例中,如果创建消息队列失败,我们将输出错误消息并退出程序。

发送消息

接下来,我们将使用msgsnd函数向消息队列发送一些消息。以下是一个发送消息的示例代码:

// sendmsg.c

#include <sys/msg.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h> typedef struct {
long type;
char text[100];
} message_t; int main()
{
key_t key = ftok("/tmp", 'a'); // 创建一个唯一的key
int msgid = msgget(key, 0666 | IPC_CREAT); // 创建消息队列
if (msgid == -1) {
perror("msgget");
exit(EXIT_FAILURE);
} message_t message;
message.type = 1;
strcpy(message.text, "Hello, World!");
int result = msgsnd(msgid, &message, sizeof(message.text), 0);
if (result == -1) {
perror("msgsnd");
exit(EXIT_FAILURE);
} printf("消息发送成功,text=%s\n", message.text); return 0;
}

在上面的代码中,我们定义了一个message_t结构体,它包含一个长整型变量和一个字符串数组。长整型变量将用于指定消息类型,而字符串数组将包含消息正文。然后,我们使用msgsnd函数将消息发送到队列。在本例中,我们发送的消息类型为1,消息正文为"Hello, World!"。

接收消息

最后,我们将使用msgrcv函数从消息队列接收我们之前发送的消息。以下是一个接收消息的示例代码:

// rsvmsg.c

#include <sys/msg.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h> typedef struct {
long type;
char text[100];
} message_t; int main()
{
key_t key = ftok("/tmp", 'a'); // 创建一个唯一的key
int msgid = msgget(key, 0666 | IPC_CREAT); // 创建消息队列
if (msgid == -1) {
perror("msgget");
exit(EXIT_FAILURE);
} message_t message;
int result = msgrcv(msgid, &message, sizeof(message.text), 1, 0);
if (result == -1) {
perror("msgrcv");
exit(EXIT_FAILURE);
} printf("消息接收成功,text=%s\n", message.text); return 0;
}

效果演示

编译上面的sendmsg.c 和 rsvmsg.c文件,得到一个两个程序:sendmsg和rsvmsg。

  • 先运行sendmsg,后运行rsvmsg
[wayne@wayne:~] ./sendmsg
消息发送成功,text=Hello, World!
[wayne@wayne:~] ./rsvmsg
消息接收成功,text=Hello, World!
  • 先运行rsvmsg,后运行sendmsg
[wayne@wayne:~] ./rsvmsg

此时rsvmsg会阻塞在这里,等待消息

[wayne@wayne:~] ./sendmsg
消息发送成功,text=Hello, World!

sendmsg发送消息后,rsvmsg进程,收到消息,打印消息

消息接收成功,text=Hello, World!

小结

总的来说,Linux 消息队列是一种高效的进程间通信机制,它可以在多个进程之间共享,允许进程异步地发送和接收消息。

以上,如果觉得对你有帮助,点个赞再走吧,这样@知微之见也有更新下去的动力!

也欢迎私信我,一起交流!

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

  1. linux 进程间消息队列通讯

    转自:http://blog.csdn.net/lifan5/article/details/7588529 http://www.cnblogs.com/kunhu/p/3608589.html 前 ...

  2. [转]Linux进程通信之POSIX消息队列

    进程间的消息队列可以用这个实现,学习了下. http://blog.csdn.net/anonymalias/article/details/9799645?utm_source=tuicool&am ...

  3. [转]Linux进程间通信——使用消息队列

    点击此处阅读原文 另收藏作者ljianhui的专栏初学Linux 下面来说说如何使用消息队列来进行进程间的通信,消息队列与命名管道有很多相似之处.有关命名管道的更多内容可以参阅我的另一篇文章:Linu ...

  4. Linux进程间通信——使用消息队列

    下面来说说如何用不用消息队列来进行进程间的通信,消息队列与命名管道有很多相似之处.有关命名管道的更多内容可以参阅我的另一篇文章:Linux进程间通信——使用命名管道   一.什么是消息队列 消息队列提 ...

  5. Linux进程通信学习总结

    http://blog.csdn.net/xiaoweibeibei/article/details/6552498 SYSV子系统的相关概念   引用标识符:引用标识符是一个整数,表示每一个SYSV ...

  6. Linux进程间通信(二) - 消息队列

    消息队列 消息队列是Linux IPC中很常用的一种通信方式,它通常用来在不同进程间发送特定格式的消息数据. 消息队列和之前讨论过的管道和FIFO有很大的区别,主要有以下两点(管道请查阅我的另一篇文章 ...

  7. Linux进程通信----匿名管道

    Linux进程通信中最为简单的方式是匿名管道 匿名管道的创建需要用到pipe函数,pipe函数参数为一个数组表示的文件描述字.这个数组有两个文件描 述字,第一个是用于读数据的文件描述符第二个是用于写数 ...

  8. linux 进程间通信之 消息队列

    消息队列就是一个消息的链表. 能够把消息看作一个记录,具有特定的格式以及特定的优先级.对消息队列有写权限的进程能够向中依照一定的规则加入新消息.有读权限的进程则能够读走消息. 读走就没有了.消息队列是 ...

  9. Linux进程内消息总线设计

    文章目录 Windows平台进程内消息总线 如果没有消息总线,会产生什么问题 死循环包含关系 高耦合.低内聚 消息总线 结构图 原理 生产者与总线的关系 总线与消费者的关系 Linux进程内消息总线设 ...

  10. Linux进程通信之System V消息队列

    System V消息队列是Open Group定义的XSI,不属于POSIX标准.System V IPC的历史相对很早,在上个世70年代后期有贝尔实验室的分支机构开发,80年代加入System V的 ...

随机推荐

  1. 文心一言 VS 讯飞星火 VS chatgpt (183)-- 算法导论13.4 7题

    七.用go语言,假设用 RB-INSERT 将一个结点 x 插入一棵红黑树,紧接着又用 RB-DELETE 将它从树中删除.结果的红黑树与初始的红黑树是否一样?证明你的答案. 文心一言: 首先,我们需 ...

  2. C语言中的操作符:了解与实践

    ​ 欢迎大家来到贝蒂大讲堂 ​ 养成好习惯,先赞后看哦~ ​ 所属专栏:C语言学习 ​ 贝蒂的主页:Betty's blog 1. 操作符的分类 操作符又叫运算符,它在C语言中起着非常大的作用,以下是 ...

  3. 275.H指数II

    1.题目介绍 给你一个整数数组 citations ,其中 citations[i] 表示研究者的第 i 篇论文被引用的次数,citations 已经按照 升序排列 .计算并返回该研究者的 h 指数. ...

  4. [转帖]linux设置page cache大小,Linux Page Cache调优在Kafka中的应用

    本文首发于 vivo互联网技术 微信公众号 链接: 作者:Yang Yijun 本文主要描述Linux Page Cache优化的背景.Page Cache的基本概念.列举之前针对Kafka的 IO ...

  5. [转帖]使用 TiUP 扩容缩容 TiDB 集群

    https://docs.pingcap.com/zh/tidb/stable/scale-tidb-using-tiup TiDB 集群可以在不中断线上服务的情况下进行扩容和缩容. 本文介绍如何使用 ...

  6. [转帖]PostgreSQL 压测工具pgbench

    1.命令 pgbench --help pgbench is a benchmarking tool for PostgreSQL. Usage:   pgbench [OPTION]... [DBN ...

  7. [转帖]实战瓶颈定位-我的MySQL为什么压不上去–写场景

    https://plantegg.github.io/2023/06/30/%E5%AE%9E%E6%88%98%E7%93%B6%E9%A2%88%E5%AE%9A%E4%BD%8D-%E6%88% ...

  8. [转帖]SQL Server数据库存储总结

    SQL Server数据库存储文件类型:数据文件和日志文件.数据文件以页面作为存储单元存储数据. 页面:即数据页面,数据页(Page).是系统在磁盘间中分配的一段大小为8k的连续空间. 文件头(Fil ...

  9. [转帖]INTEL MLC(Memory Latency Checker)介绍

    https://zhuanlan.zhihu.com/p/359823092 在定位机器性能问题的时候,有时会觉得机器莫名其妙地跑的慢,怎么也看不出来问题.CPU频率也正常,程序热点也没问题,可就是慢 ...

  10. [转帖]谈 JVM 参数 GC 线程数 ParallelGCThreads 合理性设置

    https://my.oschina.net/u/4090830/blog/7926038 1. ParallelGCThreads 参数含义 在讲这个参数之前,先谈谈 JVM 垃圾回收 (GC) 算 ...