【操作系统】进程间的通信——消息队列

本文详细介绍了Linux下消息队列的概念,包括其作为进程间通信方式的特点、限制以及操作函数msgget、msgsnd、msgrcv和msgctl的使用。通过示例展示了如何创建、发送、接收和控制消息队列,以及如何实现进程间的通信。
摘要由CSDN通过智能技术生成

进程间的通信-消息队列

什么是消息队列?

  • 消息队列,用于从一个进程向另一个进程发送数据。
  • 但仅仅把数据发送到一个"队列"中,而不指定由哪个进程来接收。
  • 消息队列独立于发送消息的进程和接收消息的进程。每个消息队列都有一个标识,只有持有这个标识的进程才可以去里面拿消息。
  • 消息队列有最大长度限制:MSGMNB。
  • 消息队列中的单条消息最大长度限制:MSGMAX。

消息队列的获取

  • msgget
  • 作用:获取或创建一个消息队列
  • 函数原型: int msgget(key_t key, int msgflg);
  • 参数:
    • key:消息队列标识。
    • msgflg:访问权限。
      • IPC_CREAT——如果key不存在,则创建,类似open函数的O_CREAT。(来源详见补充1)
      • IPC_EXCL——如果key存在,则返回失败,类似open函数的O_EXCL。(来源详见补充1)
  • 返回值:
    • 成功:正整数,即消息队列标识符。
    • 失败:返回-1,并设置errno。
  • 参考补充:

消息的发送

  • msgsnd

  • 作用:发送一个消息,即把一个消息放到某一个消息队列中

  • 函数原型:int msgsnd(int msqid, const void *msgp, size_t msgsz, int msgflg);

  • 参数:

    • msgid:消息队列标识符。

    • msgp:消息指针。

    • 消息的类型需要自己的定义。但要求其第一个结构成员为long int。

    • 例如:

    • struct  my_msg_st {
                long  int  msg_type;    /* 消息的类型,取>0, 接收消息时可使用该值 */
                            		 	 /*other info */	
      } 
      
    • msgsz:消息的长度(不包含第一个成员msg_type)

    • msgflg:

      • IPC_NOWAIT:如果包含此选项,则消息队列满时,不发送该消息,立即返回-1。反之,如果不包含此选项,则消息队列满时,挂起本进程,直到消息队列由空间可用。
  • 返回值:

    • 成功:返回0。
    • 失败:返回-1。
  • 参考补充:


消息的接收

  • msgrcv
  • 函数原型:ssize_t msgrcv (int msqid, void *msgp, size_t msgsz, long msgtype, int msgflg);
  • 功能: 从消息队列中接收一条消息
  • 参数
    • msgid: 消息队列标识符。
    • msgp: 用于接收消息的缓存。
    • msgsz:要接收的消息长度(不包括其第一个成员)
    • msgtype: 指定接收消息的类型
      • 0:从消息队列中获取第一个消息,以实现顺序接收(先发现收)。
      • 大于0:从消息队列中获取相同类型的第一个消息。
      • 小于0:从消息队列中获取消息类型小于等于msgtype绝对值的第一个消息。
    • msgflg:
      • 如果包含 IPC_NOWAIT,则当前消息队列汇总没有指定类型的消息时,立即返回-1。
      • 如果不包含 IPC_NOWAIT,则当消息队列中没有指定类型的消息时,挂起本进程,直到收到指定类型的消息。
  • 返回值:
    • 成功:返回接收到的消息的长度(不包含第一个成员msg_type)。
    • 失败: 返回-1。

消息的控制

  • msgctl
  • 函数原型:int msgctl(int msqid, int cmd, struct msqid_ds *buf);
  • 功能:用来对消息队列的基本属性进行控制、修改
  • 参数:
    • msqid:消息队列标识符。
    • cmd: 执行的控制命令。详见补充。
    • buf:详见补充。
  • 返回值:
    • 成功:返回0。
    • 失败:返回-1。
  • 补充:

示例

  • 示例1:
    • 程序1:发送消息到消息队列,程序2进行接收。

msg_send1.c

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define MSG_SIZE 80

struct my_msg_st {
	long int msg_type;
	char msg[MSG_SIZE];
};

int main(void){

	int msgid;
	int ret;
	struct my_msg_st msg;

	msgid = msgget((key_t)1235, 0666|IPC_CREAT);
	if (msgid == -1) {
		printf("msgget failed!\n");
		exit(1);
	}

	msg.msg_type = 1;	
	strcpy(msg.msg, "Hello world!");//要发送的内容
	ret = msgsnd(msgid, &msg, MSG_SIZE, 0);
	if (ret == -1) {
		printf("msgsnd failed!\n");
		exit(1);
	}

	return 0;
}

msg_recv2.c

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

#define MSG_SIZE 80

struct my_msg_st {
	long int msg_type;
	char msg[MSG_SIZE];
};

int main(void)
{
	int msgid;
	int ret;
	struct my_msg_st msg;

	msgid = msgget((key_t)1235, 0666|IPC_CREAT);
	if (msgid == -1) {
		printf("msgget failed!\n");
		exit(1);
	}

	msg.msg_type = 0;	
	ret = msgrcv(msgid, &msg, MSG_SIZE, 0, 0);
	if (ret == -1) {
		printf("msgrcv failed!\n");
		exit(1);
	}

	printf("received: %s\n", msg.msg);

	ret = msgctl(msgid, IPC_RMID, 0);//全局性的删除-IPC_RMID
	if (ret == -1) {
		printf("msgctl(IPC_RMID) failed!\n");
		exit(1);
	}

	return 0;
}

image-20220824113543268


  • 示例2:
    • 进程1:循环等待用户输入字符串,将收到的每个字符串发送给进程2,直到用户输入exit。
    • 进程2:接收进程1发来的信息,并打印输出,直到接收到exit。

msg_send.c

#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>

#define MSG_SIZE 80

struct my_msg_st {
	long int msg_type;
	char msg[MSG_SIZE];
};

int main(void){
    
	int msgid;
	int ret;
	struct my_msg_st msg;

	msgid = msgget((key_t)1235, 0666|IPC_CREAT);
	if (msgid == -1) {
		printf("msgget failed!\n");
		exit(1);
	}

	while(1) {
		fgets(msg.msg, sizeof(msg.msg), stdin);
			
		msg.msg_type = 1;	
		ret = msgsnd(msgid, &msg, MSG_SIZE, 0);
		if (ret == -1) {
			printf("msgsnd failed!\n");
			exit(1);
		}

		if (strncmp(msg.msg, "exit", 4) == 0) {
			break;
		}
	}

	return 0;
}

msg_recv.c

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

#define MSG_SIZE 80

struct my_msg_st {
	long int msg_type;
	char msg[MSG_SIZE];
};

int main(void){
    
	int msgid;
	int ret;
	struct my_msg_st msg;

    //第二个参数为权限控制
	msgid = msgget((key_t)1235, 0666|IPC_CREAT);
	if (msgid == -1) {
		printf("msgget failed!\n");
		exit(1);
	}

	while(1) {
		msg.msg_type = 0;	
		ret = msgrcv(msgid, &msg, MSG_SIZE, 0, 0);
		if (ret == -1) {
			printf("msgrcv failed!\n");
			exit(1);
		}

		printf("received: %s\n", msg.msg);

		if (strncmp(msg.msg, "exit", 4) == 0) {
			break;
		}
	}

	ret = msgctl(msgid, IPC_RMID, 0);
	if (ret == -1) {
		printf("msgctl(IPC_RMID) failed!\n");
		exit(1);
	}

	return 0;
}

image-20220824142245709

image-20220824142253700


评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

半生瓜のblog

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值