Day06(上) Liunx高级系统设计6-消息队列

概述

消息队列是消息的链表,存放在内存中,由内核维护

特点

1 、消息队列中的消息是有类型的。
2 、消息队列中的消息是有格式的。
3 、消息队列可以实现消息的随机查询。消息不一定要以先进先出的次序读取,编程时可以按消息的类型读取。
4 、消息队列允许一个或多个进程向它写入或者读取消息。
5 、与无名管道、命名管道一样,从消息队列中读出消息,消息队列中对应的数据都会被删除。
6 、每个消息队列都有消息队列标识符,消息队列的标识符在整个系统中是唯一的。
7 、只有内核重启或人工删除消息队列时,该消息队列才会被删除。若不人工删除消息队列,消息队列会一直存在于系统中
注意 :
ubuntu 某些版本中消息队列限制值如下 :
每个消息内容最多为 8K 字节
每个消息队列容量最多为 16K 字节
系统中消息队列个数最多为 1609
系统中消息个数最多为 16384

消息队列·操作命令

ipcs -q 获取消息队列信息

ipcrm -q  msgid 删除消息队列id为msgid的消息队列

获取key值

概述
System V 提供的进程间通讯机制 (IPC 通信机制 ) 需要一个 key
key 值可以是人为指定的,也可以通过 ftok 函数获得。
ftok 函数
所需头文件 :
#include <sys/types.h>
#include <sys/ipc.h>
函数 :
key_t ftok(const char *pathname, int proj_id)
参数:
        pathname:路径名 , 这个路径必须是存在的而且可以访问的
                //~ 家目录
                ///  根目录
                //./ 当前目录
        proj_id:项目 ID ,非 0 整数 ( 只有低 8 位有效 )
返回值:
        成功返回 key
        失败返回 -1
注意 :
        当调用该函数时传入的参数一致获取到的key 值也将相同
        多进程通讯时要保证路径名相同,id 也相同
示例:
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
int main(int argc, char const *argv[])
{
key_t key01 = ftok("./",2023);
key_t key02 = ftok("./",2023);
key_t key03 = ftok("./",2022);
printf("key1=%u\n",key01);
printf("key2=%u\n",key02);
printf("key3=%u\n",key03);
return 0;
}

创建和获取消息队列-msgget函数

作用:

创建一个新的或打开一个已经存在的消息队列。不同的进程调用此函数,只要用相同的
key 值就能得到同一个消息队列的标识符
函数:
所需头文件
        #include <sys/msg.h>
函数
        int msgget(key_t key, int msgflg);
参数:
        key: IPC 键值。
        msgflg:标识函数的行为及消息队列的权限。
        msgflg 的取值:
        IPC_CREAT:创建消息队列。
        IPC_EXCL:检测消息队列是否存在。
位或权限位:消息队列位或权限位后可以设置消息队列的访问权限,格式和open 函数的 mode_t 一样,但可执行权限未使用。
返回值:
        成功:消息队列的标识符(msgid) ,失败:返回 -1
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
int main(int argc, char const *argv[])
{key_t key = ftok("./",12);printf("ipc键值是:%d\n",key);int msgid = msgget(key,IPC_CREAT |0666);printf("消息队列id:%d\n",msgid);key_t key2 = ftok("./",13);int msgid2 = msgget(key2,IPC_EXCL);//查看消息队列是否存在printf("消息队列id2:%d\n",msgid2);return 0;
}

消息格式

typedef struct _msg
{
        long mtype ; /* 消息类型 , 必须是第一个成员 , 必须是 long , 就是该消息的 id*/
        char mtext [ 100 ]; /* 消息正文 , 用户自定义 */
        ... /* 消息的正文可以有多个成员 */
} MSG ;

发送消息-msgsnd函数

作用 : 将新消息添加到消息队列。
函数:
所需头文件 :
        #include <sys/msg.h>
函数 :
        int msgsnd(int msqid, const void *msgp,size_t msgsz, int msgflg);
参数 :
        msqid:消息队列的标识符。
        msgp:待发送消息结构体的地址。
        msgsz:消息正文的字节数。
        msgflg:函数的控制属性
                0:msgsnd 调用阻塞直到条件满足为止。 ( 推荐 )
                IPC_NOWAIT: 若消息没有立即发送则调用该函数的进程会立即返回。
返回值 :
        成功:0
        失败:-1

接收消息-msgrcv

作用 : 从标识符为 msqid 的消息队列中接收一个消息。一旦接收消息成功,则消息 在消息队列中被删除。
函数
所需头文件
        #include <sys/msg.h>
函数 :
        ssize_t msgrcv(int msqid, void *msgp, size_t msgsz, long msgtyp,int msgflg);
参数 :
        msqid:消息队列的标识符,代表要从哪个消息列中获取消息。
        msgp:存放消息结构体的地址。
        msgsz:消息正文的字节数。
        msgtyp:消息的类型、可以有以下几种类型
        msgtyp=0:返回队列中的第一个消息
        msgtyp>0:返回队列中消息类型为 msgtyp 的消息
        msgtyp<0:返回队列中消息类型值小于或等于 msgtyp 绝对值的消息 , 如果这种消息有若干个, 则取类型值最小的消息。
注意:
若消息队列中有多种类型的消息 ,msgrcv 获取消息的时候按消息类型获取, 不是先进先出的。
在获取某类型消息的时候 , 若队列中有多条此类型的消息 , 则获取最先添加的消息, 即先进先出原则
msgflg: 函数的控制属性
        0:msgrcv调用阻塞直到接收消息成功为止。
        MSG_NOERROR:若返回的消息字节数比 nbytes 字节数多 , 则消息就会截短到nbytes字节 , 且不通知消息发送进程
IPC_NOWAIT: 调用进程会立即返回。若没有收到消息则立即返回 -1
返回值 :
        成功返回读取消息的长度
        失败返回-1

案例1:简单的消息收发

需要 :
张三发出消息类型为 10 20 的消息
李四接收消息类型为 10 的消息
王五接收消息类型为 20 的消息
代码 02_zs.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <string.h>
typedef struct struct_msg{
long msgType;
char name[20];
char text[128];
}Msg;
int main(int argc, char const *argv[])
{
//获取key值
key_t key = ftok("./",258);
//创建或获取消息对象
int msgid = msgget(key,IPC_CREAT| 0666);
//准备要发送的消息
Msg msg01 = {10,"张三","你好 李四"};
Msg msg02;
msg02.msgType=20;
strcpy(msg02.name,"张三");
strcpy(msg02.text,"你好 王五");
//发送消息给李四
msgsnd(msgid,&msg01,sizeof(msg01) - sizeof(long),0);
//发现消息给王五
msgsnd(msgid,&msg02,sizeof(msg01) - sizeof(long),0);
return 0;
}
代码 02_ls.c
#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <string.h>
typedef struct struct_msg{
long msgType;
char name[20];
char text[128];
}Msg;
int main(int argc, char const *argv[])
{
key_t key = ftok("./",258);
int msqid = msgget(key,IPC_CREAT|0666);
Msg msg;
msgrcv(msqid,&msg,sizeof(msg)-sizeof(long),10,0);
printf("李四接收到的消息为\n");
printf("姓名:%s说%s\n",msg.name,msg.text);
return 0;
}

代码03——ww.c

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <string.h>
typedef struct struct_msg{
long msgType;
char name[20];
char text[128];
}Msg;
int main(int argc, char const *argv[])
{
key_t key = ftok("./",258);
int msqid = msgget(key,IPC_CREAT|0666);
Msg msg;
msgrcv(msqid,&msg,sizeof(msg)-sizeof(long),20,0);
printf("王五接收到的消息为\n");
printf("姓名:%s说%s\n",msg.name,msg.text);
return 0;
}

消息队列控制(了解)

作用 : 对消息队列删除 , 获取 , 修改等操作
语法 :
所需头文件 :
        #include <sys/msg.h>
函数 :
        int msgctl(int msqid, int cmd, struct msqid_ds *buf);
参数:
        msqid:消息队列的标识符。
        cmd:函数功能的控制。
                IPC_RMID:删除由 msqid 指示的消息队列,将它从系统中删除并破坏相关数据结构。                 IPC_STAT:将 msqid 相关的数据结构中各个元素的当前值存入到由 buf 指向的结构中。
                IPC_SET:将 msqid 相关的数据结构中的元素设置为由 buf 指向的结构中的对应值。
        buf:msqid_ds 数据类型的地址,用来存放或更改消息队列的属性。
返回值:
        成功: 返回 0
        失败: 返回 -1
struct msqid_ds类型:
struct msqid_ds {
struct ipc_perm msg_perm; /* 所有者与权限 */
time_t msg_stime; /* 最后一条消息发送的时间 */
time_t msg_rtime; /* 最后一条消息接收的时间 */
time_t msg_ctime; /* 最后一次更改的时间 */
unsigned long __msg_cbytes; /* 队列中的当前字节数(非标准) */
msgqnum_t msg_qnum; /* 队列中的当前消息数 */
msglen_t msg_qbytes; /* 队列中允许的最大字节数 */
pid_t msg_lspid; /* 最后一次发送的进程 id */
pid_t msg_lrpid; /* 最后一次接收的进程 id */
};
struct ipc_perm类型:
struct ipc_perm {
key_t __key; /* 提供给 msgget 的密钥 (2) */
uid_t uid; /* 所有者的有效 UID */
gid_t gid; /* 所有者的有效 GID */
uid_t cuid; /* 创建者的有效 UID */
gid_t cgid; /* 创建者的有效 GID */
unsigned short mode; /* 权限 */
unsigned short __seq; /* 序列号 */
};

案例2:多人聊天程序

#include <stdio.h>
#include <sys/types.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <unistd.h>
#include <sys/wait.h>
#include <string.h>
typedef struct INFO
{long type;char name[20];char text[128];        
}info;
int main(int argc, char const *argv[])
{printf("请输入用户姓名:\n");char name[50] = {0};scanf("%s",name);int stype = 0;printf("请输入要发送的类型:\n");scanf("%d",&stype);int rtype = 0;printf("请输入要接受的数据类型:\n");scanf("%d",&rtype);//创建消息队列key_t key = ftok("./",11) ;int mesid = msgget(key,IPC_CREAT | 0666);printf("mesid=%d\n",mesid);int i = 0;for(i = 0;i < 2;i++){int pid = fork();if(pid == 0){break;}}if(i == 0){while(1){info inf;msgrcv(mesid,&inf,sizeof(info)-sizeof(long),rtype,0);printf("%s:%s\n",inf.name,inf.text);if(strcmp("886",inf.text) == 0){break;}printf("111\n");}_exit(0);}else if(i == 1){while(1){printf("222\n");          info inf;scanf("%s",inf.text);inf.type = stype;strcpy(inf.name,name);msgsnd(mesid,&inf,sizeof(info)-sizeof(long),0);if(strcmp("886",inf.text) == 0){break;}}_exit(0);}else if(i == 2){while(1){int pid = waitpid(-1,NULL,WNOHANG);if(pid == -1){break;}}}return 0;
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/214842.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

排程系统中关于任务优先级的需求延伸与设计构思

无论是面向销售订单的MPS&#xff0c;还是基于多工序制约关系的APS&#xff0c;还是具体车间生产中针对单一工序的任务作业调度优化&#xff0c;都存在基于被排程对象(例如销售订单、生产工单、工序任务)的优先级进行优化的需求场景。当我们仅在宏观、较高层次的角度考虑&#…

[go 面试] 缓存策略与应对数据库压力的良方

关注公众号【爱发白日梦的后端】分享技术干货、读书笔记、开源项目、实战经验、高效开发工具等&#xff0c;您的关注将是我的更新动力&#xff01; 在高并发场景中&#xff0c;缓存是提高系统性能的关键利器。然而&#xff0c;缓存穿透、缓存击穿、缓存雪崩等问题可能会给系统带…

高效扫频阻垢装置广谱感应水处理设备介绍工作原理使用参数和选型

​ 1&#xff1a;高效扫频阻垢装置设备介绍 高效扫频阻垢装置是一种通过控制箱释放变频电磁信号&#xff0c;传输到信号放大装置&#xff0c;管道外侧的电磁线圈和电锤产生高频机械振动&#xff0c;在管道和水中传输&#xff0c;通过共振机理破坏水分子之间的氢键&#xff0c;产…

记录 | shell脚本开头#!/bin/bash的作用

在 Shell 脚本中&#xff0c;#!/bin/bash 是指定脚本使用 Bash 解释器的 shebang 语句。它出现在脚本的第一行&#xff0c;并告诉操作系统使用 Bash 解释器来执行该脚本 #!/bin/bash....具体作用如下&#xff1a; 指定解释器&#xff1a;#!/bin/bash 指定了使用 Bash 作为脚本…

Java - Lombok介绍、使用、工作原理、优缺点

介绍 Project Lombok is a java library that automatically plugs into your editor and build tools, spicing up your java.Never write another getter or equals method again, with one annotation your class has a fully featured builder, Automate your logging vari…

oracle修改SYS用户(系统内置超级账号)的方法和注意事项

Oracle数据库中的SYS用户是最高权限的账号&#xff0c;拥有对整个数据库的控制权。因此&#xff0c;在正常情况下&#xff0c;不建议修改SYS用户。但是有些时候为了解决特定问题&#xff0c;可能需要修改SYS用户的默认设置。 本文将介绍一些修改SYS用户的方法和注意事项。 修…

算法Day28 二进制差异序列(格雷码)

二进制差异序列&#xff08;格雷码&#xff09; Description n 位二进制差异序列是一个由2^n个整数组成的序列&#xff0c;其中&#xff1a; 每个整数都在范围[0, 2^n - 1]内&#xff08;含0和2^n - 1&#xff09; 第一个整数是0 一个整数在序列中出现不超过一次 每对相邻整数…

linux 13-2day 日志轮转 日志目录 轮转参数

目录 日志系统rsyslog一、处理日志的进程二、常见的日志文件(系统、进程、应用程序)日志优先级 三、logrotate日志轮转1、配置日志轮转的路径2、日志配置路径四 、案例 日志系统rsyslog Linux 系统内核和许多程序会产生各种错误信息、告警信息和其他的提示信息&#xff0c; 这…

【go语言实践】基础篇 - 流程控制

if语句 go里面if不需要括号将条件表达式包含起来&#xff0c;这与python也有点类似 if 条件表达式 { } if num > 18 {// ... } else if num > 20 {// ... } else {// ... }需要注意的是go支持在if的条件表达式中直接定义一个变量&#xff0c;变量的作用域只在if范围内…

【网络安全】CTF入门教程(非常详细)从零基础入门到进阶,看这一篇就够了!

一、CTF简介 CTF&#xff08;Capture The Flag&#xff09;中文一般译作夺旗赛&#xff0c;在网络安全领域中指的是网络安全技术人员之间进行技术竞技的一种比赛形式。CTF起源于1996年DEFCON全球黑客大会&#xff0c;以代替之前黑客们通过互相发起真实攻击进行技术比拼的方式。…

计算机丢失msvcp140dll怎么恢复?快速解决dll缺失问题

在计算机使用过程中&#xff0c;我们经常会遇到一些错误提示&#xff0c;其中之一就是“msvcp140dll丢失”。msvcp140.dll是一个动态链接库文件&#xff0c;它包含了许多C标准库函数的实现。这些动态链接库文件是程序运行所必需的&#xff0c;它们包含了许多函数和资源&#xf…

圣诞新奇惊喜:利用 AI 技术帮助圣诞老人创建手写信件

人工智能甚至正在接管北极的任务。在即将到来的圣诞节假期之前&#xff0c;圣诞老人和他的助手们迎来了一项革命性的技术支持。一群乐于助人的精灵采用了人工智能技术&#xff0c;制作出独一无二、看似亲手书写的信件&#xff0c;以确保遵守圣诞老人的「北极标准」。 这些信件通…

C语言实现选择排序

完整代码&#xff1a; #include<stdio.h>//交换函数&#xff0c;交换两个数 void swap(int *a,int *b){int temp;temp*a;*a*b;*btemp; }//选择排序&#xff0c;从小到大 //参数&#xff1a;arr[]表示待排序数组&#xff0c;len表示该数组长度 void select_sort(int arr[…

爱智EdgerOS之深入解析安全可靠的开放协议SDDC

一、协议简介 在 EdgerOS 的智慧生态场景中&#xff0c;许多智能设备或传感器的生命周期都与 SDDC 协议息息相关&#xff0c;这些设备可能是使用 libsddc 智能配网技术开发的&#xff0c;也有可能是因为主要功能上是使用其他技术如 MQTT、LoRa 等但是设备的上下线依然是使用上…

图的遍历(深度优先遍历 + 广度优先遍历)

目录 &#x1f33c;广度优先遍历 &#xff08;1&#xff09;邻接矩阵BFS &#xff08;2&#xff09;邻接表BFS &#xff08;3&#xff09;非连通图BFS &#xff08;4&#xff09;复杂度分析 &#x1f33c;深度优先遍历 &#xff08;1&#xff09;邻接矩阵的DFS &#x…

Caching the Application Engine Server 缓存应用程序引擎服务器

Caching the Application Engine Server 缓存应用程序引擎服务器 Application Engine caches metadata just like the application server. This caching enhances performance because a program can refer to the local cache for any objects that it uses. 应用程序引擎…

科技云报道:从数据到生成式AI,是该重新思考风险的时候了

科技云报道原创。 OpenAI“宫斗”大戏即将尘埃落定。 自首席执行官Sam Altman突然被董事会宣布遭解雇、董事长兼总裁Greg Brockman辞职&#xff1b;紧接着OpenAI员工以辞职威胁董事会要求Altman回归&#xff1b;再到OpenAI董事会更换成员、Altman回归OpenAI。 表面上看&…

深入解析Java中的String:特点、重要方法及源码分析

Java的String类是Java语言中最常用的类之一。 作为一位Java高级工程师&#xff0c;了解String类的特性和方法对于编写高效和优化的Java代码至关重要。在这篇技术博客中&#xff0c;我们将深入探讨String类的特点&#xff0c;介绍其中一些重要的方法&#xff0c;并分析其源码以获…

java--LocalDate、LocalTime、LocalDateTime、ZoneId、Instant

1.为什么要学习JDK8新增的时间 LocalDate&#xff1a;代表本地日期(年、月、日、星期) LocalTime&#xff1a;代表本地时间(时、分、秒、纳秒) LocalDateTime&#xff1a;代表本地日期、时间(年、月、日、星期、时、分、秒、纳秒) 它们获取对象的方案 2.LocalDate的常用API(…

Android的开机logo生成

生成可用的uboot和kernel的logo图片 可以通过命令转换BMP格式的图片 ### 将 png 转为颜色深度为8bit的的bmp图片。jpeg使用jpegtopnm ### pngtopnm logo.png | ppmquant 31 | ppmtobmp -bpp 8 > logo.bmp然后就可以使用新图替换旧图片&#xff0c;在kernel目录下的logo.bmp…