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


    进程间的通信-消息队列

    什么是消息队列?

    • 消息队列,用于从一个进程向另一个进程发送数据。
    • 但仅仅把数据发送到一个"队列"中,而不指定由哪个进程来接收。
    • 消息队列独立于发送消息的进程和接收消息的进程。每个消息队列都有一个标识,只有持有这个标识的进程才可以去里面拿消息。
    • 消息队列有最大长度限制: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 */	
        } 
        
        • 1
        • 2
        • 3
        • 4
      • 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 
    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    
    #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;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38

    msg_recv2.c

    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    
    #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;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44

    image-20220824113543268


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

    msg_send.c

    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    
    #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;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45

    msg_recv.c

    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    #include 
    
    #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;
    }
    
    • 1
    • 2
    • 3
    • 4
    • 5
    • 6
    • 7
    • 8
    • 9
    • 10
    • 11
    • 12
    • 13
    • 14
    • 15
    • 16
    • 17
    • 18
    • 19
    • 20
    • 21
    • 22
    • 23
    • 24
    • 25
    • 26
    • 27
    • 28
    • 29
    • 30
    • 31
    • 32
    • 33
    • 34
    • 35
    • 36
    • 37
    • 38
    • 39
    • 40
    • 41
    • 42
    • 43
    • 44
    • 45
    • 46
    • 47
    • 48
    • 49
    • 50
    • 51

    image-20220824142245709

    image-20220824142253700


  • 相关阅读:
    【计算机视觉 | 语义分割】语义分割常用数据集及其介绍(七)
    40. 干货系列从零用Rust编写负载均衡及代理,websocket的实现
    【校招VIP】产品行测能力之数图问题
    某评登录与数据获取
    【Scala入门学习】基本数据类型和变量声明
    使用Hadoop MapReduce分析邮件日志提取 id、状态 和 目标邮箱
    字符串编辑距离
    提高APP安全性的必备加固手段——深度解析代码混淆技术
    Java并发编程第11讲——AQS设计思想及核心源码分析
    【Android笔记30】Android中数据存储之SQLite数据库
  • 原文地址:https://blog.csdn.net/qq_51604330/article/details/126504062