Linux进程通信 | 消息队列
创始人
2025-05-29 19:02:43

什么是消息队列?

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

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

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

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

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

消息队列的优点和缺点

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

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

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

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

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

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

#include 
#include 
#include 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 
#include 
#include int main()
{key_t key = ftok("/tmp", 'a'); // 创建一个唯一的keyint 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 
#include 
#include 
#include typedef struct {long type;char text[100];
} message_t;int main()
{key_t key = ftok("/tmp", 'a'); // 创建一个唯一的keyint 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 
#include 
#include 
#include typedef struct {long type;char text[100];
} message_t;int main()
{key_t key = ftok("/tmp", 'a'); // 创建一个唯一的keyint 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 消息队列是一种高效的进程间通信机制,它可以在多个进程之间共享,允许进程异步地发送和接收消息。

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

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

相关内容

热门资讯

全市场近3400股下跌 全市场... 2025.06.10本文字数:1222,阅读时长大约2分钟作者 |第一财经6月10日,截至午间收盘,...
从李蓓总的新文说起,建筑奇观为... 请你提供李蓓总的新文具体内容呀,没有相关内容我没法准确进行描述呢。你可以简单说说新文的大概主题、观点...
曝国资和员工利益受损,娃哈哈股... 娃哈哈股权再现争议,该如何解决?上个月娃哈哈纯净水被曝由今麦郎代工的风波尚未平息。娃哈哈又在最近又迎...
苹果开了 WWDC,股价跌了:... 大数据文摘出品苹果 WWDC 发布会开了,但股票市值下跌了。初步估算,按苹果3.01 万亿美元市值计...
史上“最无聊”的WWDC结束后... 本文来源:时代周报 作者:谢斯临 图片来源:WWDC直播截图这可能是苹果(AAPL.NASDAQ)...
恒生指数半日涨0.33% 恒生... 【恒生指数半日涨0.33%】截至午间收盘,恒生指数涨0.33%,恒生科技指数跌0.33%,恒生中国企...
东阳光药与唯信计算达成战略合作... (图片来源:视觉中国) 东阳光药&唯信计算 近日,广东东阳光药业股份有限公司(以下简称东阳光药)与唯...
陈春花:AI时代企业家能力跃迁... 在人工智能重塑商业文明的时代浪潮中,企业家正经历从技术适应者到生态构建者的根本跃迁。这场变革不仅要求...
恒生ETF:6月9日融资买入1... 证券之星消息,6月9日,恒生ETF(513660)融资买入1174.0万元,融资偿还1398.34万...
俄卢布反弹超40%,美元“前所... 在地缘政治局势紧张叠加油价下跌的背景下,俄罗斯卢布却展现出惊人的反弹。 近日,美国银行数据显示,卢...
药易购:6月9日融资买入709... 证券之星消息,6月9日,药易购(300937)融资买入709.6万元,融资偿还863.31万元,融资...
科大讯飞董事长刘庆峰司庆讲话透... 6月9日,科大讯飞成立26周年之际,董事长刘庆峰发表内部讲话,阐述公司技术创新逻辑和战略定力。 在近...
思源电气跌1.41%,成交额3... 6月9日,思源电气跌1.41%,成交额3.16亿元,换手率0.72%,总市值562.27亿元。 异动...
派林生物拟易主中国生物,整合预... 6月10日,派林生物(000403.SZ)股票恢复交易,盘中股价涨幅超过3%。前一晚,派林生物发布了...
重大资产重组!中科曙光,一字涨... 作 者丨 张赛男 雷晨 彭卓编 辑丨 骆一帆 江佩佩6月10日,中科曙光复牌一字涨停,封单金额超16...
杜嘉祺重返友邦保险接任董事长:... 2025年6月6日,友邦保险宣布了一项重大人事变动:董事会主席谢仕荣在任职60年后正式退任,被授予荣...
碧桂园:市场正朝着积极的方向发... 6月9日,碧桂园(02007.HK)召开月度管理会。会上,董事会主席杨惠妍表示,“在各方的共同努力下...
招商蛇口前5个月销售额约671... 招商蛇口 视觉中国 资料图6月9日晚间,招商蛇口(001979.SZ)披露前5个月销售情况。数据显...
DeepSeek概念股深度回调... 6月10日,DeepSeek概念股深度回调,深信服跌逾5%,泛微网络、金山办公跌逾4%。
千亿重组启动!海光信息拟116... 一场或将重塑中国算力产业版图的千亿级资本重组大幕正徐徐拉开。6月10日,国内服务器巨头中科曙光(60...
如何看待“人才交流型并购” “人才交流型并购”是一种颇具战略意义的并购模式。它不仅仅是企业资产的整合,更重要的是人才的流动与融合...
谁在定义明天的家?极有家十周年... “逛个街居然被种草整套家具。”陈悦在社交平台推荐的上海Citywalk路线上,意外被翡悦里的家装展吸...
港股创新药50ETF(5137... 6月10日,创新药概念再度走高。相关ETF方面,港股创新药50ETF(513780)盘中涨超4%,今...
北京五八到家信息技术集团有限公... 国家企业信用信息公示系统网站显示,6月6日,北京五八到家信息技术集团有限公司因通过登记的住所或者经营...
角逐稳定币,美元化卷土重来?|... 各国家地区前所未有的立法背书,使稳定币获得了比前辈比特币更高规格的发展空间。其将冲击传统跨境支付,这...
极有家发布2025中国家装七大... 十年前,淘宝家装品类开始蓬勃发展。彼时家装行业面临非标产品缺失信任、行业标准空白等痛点,消费者对"好...
3400点得而复失 震荡向上的... 解锁【第一财经智享会员】实时解读市场动态,把握投资先机。【第一财经智享会员专属】晨会博弈【今日早盘】...
我国外汇储备规模上升 央行连续... 中国商报(记者 马文博)国家外汇管理局最新数据显示,截至2025年5月末,我国外汇储备规模为3285...
撬动万亿文旅市场,美团的底牌是... 导语:玩得深、走得远、舍得花的年轻人,不只活在酒店的房晚数据里,更活跃在本地生活的高频行为中。“一业...