linux应用开发基础知识(七)——管道和消息队列进程通信

管道通信

匿名管道

#include  <unistd.h>
int  pipe(int pfd[2]);

pfd[0]用于读管道,而pdf[1]用于写管道。
注意:匿名管道只能用于亲缘关系的进程之间通信。管道通道是单向的,一边读,另一边写。管道可以用于大于两个进程共享。
例如设计父进程读,两个子进程写的代码如下:

#include<stdio.h>
#include<pthread.h>
#include<stdlib.h>
#include<unistd.h>
#include<string.h>
int main(int argc, char* argv)
{int pipegd[2];int i;char buf[19]={0};pid_t pid;pid_t pid1;int pi = pipe(pipegd);if(pi<0)
{perror("pipe:");return -1;
}
pid=fork();
if(pid<0){perror("pid create:");return 0;}
else if(pid == 0){close(pipegd[0]);memset(buf, 0, 19);strcpy(buf, "this is b");write(pipegd[1], buf, strlen(buf));}else if(pid>0){pid1=fork();if(pid1<0){perror("pid create:");return 0;}else if(pid1 == 0){close(pipegd[0]);memset(buf, 0, 19);strcpy(buf, "this is a");write(pipegd[1], buf, strlen(buf));}else if(pid1>0){sleep(1);close(pipegd[1]);memset(buf,0,19);read(pipegd[0],buf,19);printf("pipe read is \"%s\"\n",buf);}}
return 0;
}

有名管道

有名管道也叫命名管道。
1、管道文件仅仅是文件系统中的标示,并不在磁盘上占据空间。在使用时,在内存上开辟空间,作为两个进程数据交互的通道。也就是提供一个路径名与之关联,这样,即使与创建有名管道的进程不存在亲缘关系的进程,只要可以访问该路径,就能够通过这个有名管道进行相互通信。
2、并且顾名思义,管道秉承着“先进先出”的原则。先写进去的数据,会被先读出来。
3、并且管道需要读进程先存在,否则写进程是没有什么意义的。
有名管道通信的API

#include  <unistd.h>
#include <fcntl.h>
//创建管道的头文件
#include <sys/types.h>
#include <sys/stat.h>
int  mkfifo(const char *path, mode_t mode);
open(const char *path, O_RDONLY);//1
open(const char *path, O_RDONLY | O_NONBLOCK);//无阻塞读
open(const char *path, O_WRONLY);//3
open(const char *path, O_WRONLY | O_NONBLOCK);//无阻塞写

并且需要注意
1)使用open函数打开管道文件
如果一个进程以只读(只写)打开,那么这个进程会被阻塞到open,直到另一个进程以只写(只读)或者读写。
2)使用read函数读取内容
read读取普通文件,read不会阻塞。而read读取管道文件,read会阻塞运行,直到管道中有数据或者所有的写端关闭。
利用上面的API实现父子进程通信:子进程写,父进程读

#include"stdio.h"
#include"unistd.h"
#include"string.h"
#include"fcntl.h"
#include <sys/types.h>
#include <sys/stat.h>
int main(int argc,char* argv[])
{int pd,fd1,fd2;char buf[20]={0};char buf1[20]="hello,world!";char str[]="./my";pid_t pid;pd = mkfifo(str,S_IFIFO|0666);if(pd<0){perror("mkfifo:");return -1;}pid = fork();if(pid<0){perror("pthread creat");}if(pid==0){fd2 = open(str,O_WRONLY);if(fd2<0){perror("write:");return -1;}write(fd2,buf1,20);}else{fd1 = open(str,O_RDONLY); if(fd1<0){perror("read:");return -1;}read(fd1,buf,20);printf("read = %s\n",buf);}
return 0;
}

创建两个进程来实现非亲缘关系进程通信
read部分

#include"stdio.h"
#include"string.h"
#include"unistd.h"
#include"sys/types.h"
#include"sys/stat.h"
#include <fcntl.h>
int main(int argc,char* argv[])
{int pd,fd;pid_t pid;char str[]="./mypipe";char buf[]="you!";// pd = mkfifo(str,0666);// if(pd<0)// {//     perror("mkfifo");//     return -1;// }fd = open(str,O_WRONLY);if(fd<0){perror("open");return -1;}while(1){int ret = write(fd,buf,sizeof(buf));if(ret<0){perror("write");return -1;}printf("write successful!");printf("write buf is %s",buf);sleep(1);}return 0;
}

write部分

#include"stdio.h"
#include"string.h"
#include"unistd.h"
//创建管道头文件
#include"sys/types.h"
#include"sys/stat.h"
//下面的大写宏定义头文件
#include <fcntl.h>
int main(int argc,char* argv[])
{int pd,fd;pid_t pid;char str[]="./mypipe";char buf[30];fd = open(str,O_RDONLY);if(fd<0){perror("open");return -1;}read(fd,buf,30);printf("read is:%s\n",buf);return 0;
}

需要两边都开着才行,不然会进行堵塞。

总结就是:
(1)两个进程运行时,写端彻底关闭,则读端read返回0,也会关闭。
(2)如果管道的读端关闭,继续写入数据会发生异常,写端收到未捕获的信号SIGPIPE会关闭写端。当然也可以修改默认的信号响应方式,比如增加信号处理函数。
(3)写端写入数据以后,读端不从里面读取内容:数据保持在管道中存在一段时间。管道文件的大小是0。
(4)管道通讯发送的数据若没有指定进程接收,任何一个进程只要打开的是同一个管道文件,都有可能读到数据。
(5)read读取管道中的数据,只要读过的数据就会被清空 。

消息队列

消息队列的本质就是存放在内存中的消息的链表,而消息本质上是用户自定义的数据结构。如果进程从消息队列中读取了某个消息,这个消息就会被从消息队列中删除。
对比一下管道机制:

1、消息队列允许一个或多个进程向它写入或读取消息。
2、消息队列可以实现消息的随机查询,不一定非要以先进先出的次序读取消息,也可以按消息的类型读取。比有名管道的先进先出原则更有优势。
3、对于消息队列来说,在某个进程往一个队列写入消息之前,并不需要另一个进程在该消息队列上等待消息的到达。而对于管道来说,除非读进程已存在,否则先有写进程进行写入操作是没有意义的。
4、消息队列的生命周期随内核,如果没有释放消息队列或者没有关闭操作系统,消息队列就会一直存在。而匿名管道随进程的创建而建立,随进程的结束而销毁。

需要注意的是,消息队列对于交换较少数量的数据很有用,因为无需避免冲突。但是,由于用户进程写入数据到内存中的消息队列时,会发生从用户态拷贝数据到内核态的过程;同样的,另一个用户进程读取内存中的消息数据时,会发生从内核态拷贝数据到用户态的过程。因此,如果数据量较大,使用消息队列就会造成频繁的系统调用,也就是需要消耗更多的时间以便内核介入。

消息队列的一些API

 #include <sys/ipc.h>#include <sys/msg.h>key_t ftok(const char *pathname, int proj_id);把一个已存在的路径名和一个整数标识符转换成IPC键值int msgget(key_t key, int msgflg);创建一个消息队列的函数int msgsnd(int msgid, const void *msgp, size_t size, int msgflg);向这个消息队列发送消息int msgrcv(int msgid, void *msgp, size_t size, long msgtype, int msgflg);向这个消息队列读取消息int msgctl(int msgid, int cmd, struct msqid_ds *buf);消息队列的控制

对这些API的解释:

key_t ftok(const char *pathname, int proj_id);
把一个已存在的路径名和一个整数标识符转换成IPC键值

1、成功时返回返回key_t值(即IPC 键值),失败时返回-1;
2、pathname:指定的文件,此文件必须存在且可存取; proj_id:计划代号(project ID)。
3、在使用ftok()函数时,里面有两个参数,即fname和id,fname为指定的文件名,而id为子序列号,这个函数的返回值就是key,它与指定的文件的索引节点号和子序列号id有关,但是如果文件被删除然后重新创建了一个一样的,这个key值也会变的。

 int msgget(key_t key, int msgflg);创建一个消息队列的函数

1、成功时返回消息队列的id,失败时返回EOF(-1)
2、key和消息队列关联的key IPC_PRIVATE 或 ftok
3、msgflg 标志位: IPC_CREAT|0666,IPC_CREAT:没有创建则创建,有则打开。

int msgsnd(int msgid, const void *msgp, size_t size, int msgflg);
向这个消息队列发送消息

1、成功时返回0,失败时返回-1
2、msgid:消息队列id;msgp: 消息缓冲区地址;size: 消息正文长度
3、msgflg:标志位 0 或 IPC_NOWAIT
0: 当消息队列满时,msgsnd将会阻塞,直到消息能写进消息队列
IPC_NOWAIT: 当消息队列已满的时候,msgsnd函数不等待立即返回

int msgrcv(int msgid, void *msgp, size_t size, long msgtype, int msgflg);
向这个消息队列读取消息

1、成功时返回收到的消息长度,失败时返回-1
2、msgid : 消息队列id;msgp : 消息缓冲区地址;size : 指定接收的消息长度;msgtype : 指定接收的消息类型;
msgtype=0: 收到的第一条消息,任意类型。
msgtype>0: 收到的第一条 msg_type类型的消息。
msgtype<0: 接收类型等于或者小于msgtype绝对值的第一个消息。
例子:如果msgtype=-4,只接受类型是1、2、3、4的消息
3、msgflg : 标志位
0: 阻塞式接收消息
IPC_NOWAIT: 如果没有返回条件的消息调用立即返回,此时错误码为ENOMSG
MSG_EXCEPT: 与msgtype配合使用返回队列中第一个类型不为msgtype的消息

 int msgctl(int msgid, int cmd, struct msqid_ds *buf);消息队列的控制

1、成功时返回0,失败时返回-1
2、msgid : 消息队列id;cmd : 要执行的操作 IPC_STAT/ IPC_SET / IPC_RMID(删除)
3、buf : 存放消息队列属性的地址

创建两个.c文件来实现消息队列的写入和读取。

写入消息队列

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdlib.h>
#define LEN (sizeof(MSG) - sizeof(long))
typedef struct
{long mtype;char mtext[64];
}MSG;
int main(int argc,char *argv[])
{MSG buf;key_t key;int msgid,mssend,msrecv;//fgets(buf.mtext,64,stdin);if((key = ftok(".",'q')) == -1){perror("ftok");return -1;}if(msgid = msgget(key,IPC_CREAT|0666)<0){perror("magid");return -1;}buf.mtype = 1;strcpy(buf.mtext, "this is message 1");if(mssend = msgsnd(msgid,buf.mtext,LEN,0)<0){perror("mssend");return -1;}buf.mtype = 2;strcpy(buf.mtext, "this is message 2");if(mssend = msgsnd(msgid,buf.mtext,LEN,0)<0){perror("mssend");return -1;}buf.mtype = 3;strcpy(buf.mtext, "this is message 3");if(mssend = msgsnd(msgid,buf.mtext,LEN,0)<0){perror("mssend");return -1;}return 0;
}

读取消息队列

#include <stdio.h>
#include <string.h>
#include <unistd.h>
#include <sys/ipc.h>
#include <sys/msg.h>
#include <stdlib.h>
#define LEN (sizeof(MSG) - sizeof(long))
typedef struct
{long mtype;char mtext[64];
}MSG;
int main(int argc,char *argv[])
{MSG buf;key_t key;int msgid,msrecv;buf.mtype = 1;int count = 0;//fgets(buf.mtext,64,stdin);if((key = ftok(".",'q')) == -1){perror("ftok");return -1;}if(msgid = msgget(key,IPC_CREAT|0666)<0){perror("magid");return -1;}while(1){if(msrecv = msgrcv(msgid,&buf,64,0,0) < 0){perror("msrecv");return -1;}printf("msrecv = %s\n",buf.mtext);count++;if(count==3)break;}return 0;
}

最后结果

在这里插入图片描述

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

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

相关文章

怎么使用python进行整除取余求幂

怎么使用python进行整除取余求幂&#xff1f; 整除法是//&#xff0c;称为地板除&#xff0c;两个整数的除法仍然是整数。 10//33 3 求模运算是%&#xff0c;相当于mod&#xff0c;也就是计算除法的余数。 5%2 1 求幂运算使用两个连续的*&#xff0c;幂运算符比取反的优先级高…

2024 最新推广服务 API 推荐,助力业务腾飞

在数字化营销的浪潮中&#xff0c;API 服务正以其强大的功能和高效的特性&#xff0c;成为企业和开发者们实现精准推广、优化营销效果的得力助手。2024 年的今天&#xff0c;各种创新的 API 服务层出不穷&#xff0c;为广告投放、数据洞察等领域带来了前所未有的机遇。在接下来…

数字设计的秘密:原来有这么多创意玩法!

数字产品设计涵盖了各种学科和角色&#xff0c;但主要是人与数字设备或数字界面之间的互动。数字设计的两个主要部分是用户体验设计&#xff08;UX&#xff09;和用户界面设计&#xff08;UI&#xff09;。如果把数字产品的设计想象成人体&#xff0c;那么骨架就是代码&#xf…

从0-1搭建一个web项目vue3+vite+ts+element-plus(脚手架分析)

本章分析从0-1的搭建脚手架依赖 ObJack-Admin一款基于 Vue3.3、TypeScript、Vite3、Pinia、Element-Plus 开源的后台管理框架。在一定程度上节省您的开发效率。另外本项目还封装了一些常用组件、hooks、指令、动态路由、按钮级别权限控制等功能。感兴趣的小伙伴可以访问源码点个…

结构体(二)

今天来继续介绍我们有关结构体的相关知识 结构体的自引用 结构体的自引用&#xff0c;顾名思义嘛&#xff0c;就是在我们的结构体中再次引用该结构体&#xff0c;这一点跟我们的函数递归有异曲同工之妙&#xff0c;不了解函数递归的小伙伴可以移步到我之前做过的一期&#xf…

GPTCache:革新大模型缓存,降低成本,提升效率

GPTCache介绍 随着应用程序越来越受欢迎并遇到更高的流量水平,与 LLM API 调用相关的费用可能会变得相当可观。此外,LLM 服务的响应时间可能会很慢,尤其是在处理大量请求时。GPTCache是一个致力于构建用于存储 LLM 响应的语义缓存的项目。 项目架构 数字人助力传统客服 1…

效率神奇分享

嘿宝贝们&#xff01;&#x1f44b; 知道吗&#xff1f;工作效率高不高&#xff0c;其实跟你用的工具有很大关系哦&#xff01;今天小红书种草专家来给大家安利五款超实用的国产工作app&#xff0c;让你的工作生活轻松又高效&#xff01;&#x1f680; 1️⃣ 【亿可达】&#…

植物大战僵尸杂交版技巧大全(附下载攻略)

《植物大战僵尸杂交版》为策略游戏爱好者带来了全新的挑战和乐趣。如果你是新手玩家&#xff0c;可能会对游戏中的植物和僵尸感到困惑。以下是一些实用的技巧&#xff0c;帮助你快速掌握游戏并享受其中的乐趣。 技巧一&#xff1a;熟悉基本玩法 游戏的基本玩法与原版相似&…

【Android】多种方式实现截图(屏幕截图、View截图、长图)

目录 一、截图原理二、实现方式1. View截图2. WebView截图3. 屏幕截图 三、格式转换方法 一、截图原理 我们的手机一般同时按下音量-键和电源键就会将当前屏幕显示的内容截取下来&#xff0c;那里面具体经过哪些流程呢&#xff1f; Android中每一个页面都是一个Activity&#…

Java洗鞋小程序预约系统源码

&#x1f4a5;洗鞋神器来袭&#xff01;轻松预约&#xff0c;让你的鞋子焕然一新&#x1f45f; &#x1f389; 告别洗鞋烦恼&#xff0c;洗鞋预约小程序来啦&#xff01; 你是不是常常为洗鞋而烦恼&#xff1f;手洗太累&#xff0c;送去洗衣店又贵又麻烦。现在&#xff0c;好…

std::enable_if和std::is_base_of

std::enable_if,其主要为了完成模板特偏化&#xff0c;有两个参数&#xff0c;第一个为布尔值类型&#xff0c;第二个如果布尔值为true&#xff0c;其为默认空值&#xff0c;如果已经赋值&#xff0c;则为对应的类型。 std::is_base_of&#xff0c;其一共存在两个参数&#xff…

外汇的基本面分析需要关注什么?

外汇基本面分析的核心在于关注可能影响单一货币供求及国家货币价值的经济、社会和地缘政治事件与趋势。但值得注意的是&#xff0c;这些事件和因素往往具有更广泛的影响力&#xff0c;不仅限于单一国家。它们可能是影响整个地区或国家集团的重要事件&#xff0c;甚至一些事件&a…

数学建模 —— MATLAB中的矩阵(下)

目录 矩阵的拼接 矩阵的重复 矩阵的重构和重新排列 (1)reshape 函数 (2)sort 函数 (3)sortrows 函数 (4)flip / fliplr / flipud 函数 (5)rot90 函数 矩阵的拼接 有时候我们需要对多个矩阵进行拼接&#xff0c;变成一个大的矩阵。根据矩阵拼接的方向&#xff0c;我们可 …

Java实现RS485串口通信

博客链接地址 近期&#xff0c;我接到了一个任务&#xff0c;将报警器接入到Java项目中&#xff0c;而接入的方式就是通过RS485接入&#xff0c;本人之前可以说是对此毫无所知。不过要感谢现在的互联网&#xff0c;通过网络我查到了我想要知道的一切&#xff0c;这里记录下本次…

STM32音频应用开发:DMA与定时器的高效协作

摘要: 本文章将深入浅出地介绍如何使用STM32单片机实现音频播放功能。文章将从音频基础知识入手&#xff0c;逐步讲解音频解码、DAC转换、音频放大等关键环节&#xff0c;并结合STM32 HAL库给出具体的代码实现和电路设计方案。最后&#xff0c;我们将通过一个实例演示如何播放W…

揭示优化Prompt的秘诀:如何让API表现媲美网页版

为什么用GPT API&#xff08;GPT-3.5-turbo&#xff09;进行程序分析时&#xff0c;效果好像比网页版的GPT-3.5差一点&#xff1f;这可能有几个原因&#xff0c;咱们细说一下。 1. Prompt不同 这是最常见的问题之一。API调用时的指令&#xff08;prompt&#xff09;往往比较简…

Ceil()——向上取整函数

函数原型为&#xff1a; double ceil(double x); 大家可以在这个网站里更清晰的了解ceil - C Reference (cplusplus.com) 下面借助一道例题来帮助大家理解&#xff1a;牛牛的快递_牛客题霸_牛客网 (nowcoder.com) 我们分析题得知&#xff0c;在大于1的情况下&#xff0c;只要…

【ocean】ocnPrin结合getData导出数据

核心就是这一句ocnPrint(?output fout leafValue( getData(“/output” ?result “dc”))) r_list list(4000, 4100, 4200) multi_list list(20,21,22) fout outfile("/home/yourpath/results.txt" "w") foreach(r_value r_listforeach(multi_value …

【unity实战】制作unity数据保存和加载系统——大型游戏存储的最优解

最终效果 文章目录 最终效果前言存储位置信息存储更多数据存储场景信息持久化存储数据完结 前言 前面写过小型游戏存储功能&#xff1a; 【unity实战】制作unity数据保存和加载系统——小型游戏存储的最优解&#xff08;包含数据安全处理方案的加密解密&#xff09; 这次做一…

基于RK3568车载电脑助力日本巴士公司高效完成巴士到站系统项目部署

无处不在的物联网&#xff08;IoT&#xff09;技术已经渗透到了人类生活的各个角落&#xff0c;如日常出行乘坐的公交车上&#xff0c;物联网&#xff08;IoT&#xff09;技术的应用就得到完美诠释&#xff01;其通过公交车上的车载电脑网络与中控室服务器连接来对公交车的运行…