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

news2024/11/20 14:20:15

管道通信

匿名管道

#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.coloradmin.cn/o/1866863.html

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

相关文章

基线核查--渗透

基线检查 基线核查概念 it中定义&#xff1a; 基线为初始的标准&#xff0c;以后更改就要经过授权&#xff0c;形成下一基线。 软件配置管理的基线&#xff1a;1功能基线&#xff0c;分配基线&#xff0c;产品基线 安全配置基线--基线核查 安全基线可以说是木桶理论&…

Dubbo运行原理

目录 Dubbo通讯协议 Dubbo负载均衡策略 RPC和HTTP有什么区别&#xff1f; 让你设计一个RPC框架&#xff0c;如何考虑数据序列化问题&#xff1f; Dubbo 是一款高性能、轻量级的开源 RPC&#xff08;远程过程调用&#xff09;框架&#xff0c;主要用于构建分布式服务和微服务…

为什么 Uvicorn 的性能不如 uWSGI?你真的用对了吗?

Uvicorn 简介 Uvicorn 是一个基于 ASGI 的快速 Web 服务器,号称性能赶超 uWSGI。然而,有些人在实际使用中发现 Uvicorn 的性能不如 uWSGI。那么,Uvicorn 真的不如 uWSGI 吗?其实,问题可能出在使用方法上。 Uvicorn 是否真的不如 uWSGI 首先,我们需要了解 Uvicorn 和 uW…

无源电压继电器 JDY-1210AW 导轨安装 约瑟JOSEF

系列型号&#xff1a; JDY-1002AW电压继电器&#xff1b;JDY-1002B电压继电器&#xff1b; JDY-1110AW电压继电器&#xff1b;JDY-1110B电压继电器&#xff1b; JDY-1220AW电压继电器&#xff1b;JDY-1220B电压继电器&#xff1b; JDY-1100AW电压继电器&#xff1b;JDY-110…

浅谈业务开发与非业务开发

浅谈业务开发与非业务开发 软件开发业务开发非业务开发工作量的区别 软件开发 在谈及业务开发与非业务开发之前&#xff0c;首先他们都是软件开发&#xff0c;那么软件开发的流程是怎样的呢&#xff1f;我们先来了解一下软件开发的流程。通常情况下软件开发的流程是这样的 在…

Python中20个鲜为人知的字符串函数

目录 1. capitalize() 2. casefold() 3. join() 和 split() 4. strip(), lstrip(), rstrip() 5. replace() 6. format() 7. enumerate() 8. isalpha(), isdigit(), isalnum() 9. startswith(), endswith() 10. center() 11. count() 12. find(), index() 13. make…

【SQL】数据操作语言(DML) - 删除数据:精细管理数据的利刃

目录 前言 DELETE语句的基础使用 删除指定记录 清空表与删除表数据的区别 注意 前言 在数据库管理的日常工作中&#xff0c;数据的删除是一项需要格外小心的操作&#xff0c;因为一旦数据被删除&#xff0c;往往难以恢复。数据操作语言(DML)中的DELETE语句&am…

MFC---静态文本框和编辑框Edit Control(控件的通知消息)(常用控件)

上一节讲了颜色对话框之后&#xff0c;关于对话框的使用和各种通用对话框的介绍就到此为止了。从本节开始将讲解各种常用控件的用法。常用控件主要包括&#xff1a;静态文本框、编辑框、单选按钮、复选框、分组框、列表框、组合框、图片控件、列表控件、树形控件和进度条控件等…

苹果笔记本双系统怎么安装

想要在mac电脑上装双系统&#xff0c;首先需要确认您的电脑是否支持。苹果电脑自带的boot camp工具可以帮助您在mac上安装windows系统&#xff0c;只需按照步骤进行操作即可。另外&#xff0c;您也可以使用虚拟机软件&#xff0c;如parallels desktop或vmware fusion&#xff0…

uniapp 微信小程序端使用百度地图API

1、登录百度地图开放平台 https://lbsyun.baidu.com/&#xff08;没有账号则先去创建一个百度账号&#xff09; 2、进入百度地图开放平台控制台&#xff08;导航栏“控制台”&#xff09;&#xff0c;点击“应用管理”-“我的应用” 3、选择“创建应用”&#xff0c;应用模块选…

前端开发实战项目:实时天气预报应用

引言 在本实战项目中&#xff0c;我们将开发一个实时天气预报应用。这个项目将帮助你掌握前端开发的核心技能&#xff0c;包括HTML、CSS、JavaScript&#xff0c;以及如何使用API来获取实时数据。通过这个项目&#xff0c;你将学会如何构建用户界面、处理用户交互、以及与第三…

Java知识点整理 13 — Hutool工具库

在开发时经常需要编写很多与业务无关的代码&#xff0c;比如获取指定日期对象、获取本机 IP 地址、数据加密等。通常我们会将这些代码独立出来&#xff0c;放到 utils 目录下&#xff0c;作为工具类供其它代码调用。 但如果遇到一个从未接触过的领域知识&#xff0c;开发一个新…

python-如何将Python 脚本打包成可执行文件(exe)

文章目录 前言如何将Python 脚本打包成可执行文件&#xff08;exe&#xff09;1. 测试python脚本2. 安装 PyInstaller3. 创建 PyInstaller spec 文件4. 生成可执行文件4.1 去掉黑框 前言 如果您觉得有用的话&#xff0c;记得给博主点个赞&#xff0c;评论&#xff0c;收藏一键三…

【论文阅读】--Popup-Plots: Warping Temporal Data Visualization

弹出图&#xff1a;扭曲时态数据可视化 摘要1 引言2 相关工作3 弹出图3.1 椭球模型3.1.1 水平轨迹3.1.2 垂直轨迹3.1.3 组合轨迹 3.2 视觉映射与交互 4 实施5 结果6 评估7 讨论8 结论和未来工作致谢参考文献 期刊: IEEE Trans. Vis. Comput. Graph.&#xff08;发表日期: 2019&…

基于百度地图实现矩形绘制/电子围栏/自定义覆盖物选择、点击、区域选中、轨迹绘制

目录 开发前的准备账号注册页面创建地图初始化矩形绘制开启绘制模式监听绘制完成事件矩形取消事件自定义覆盖物渲染数据准备覆盖物渲染自定义点击事件优化用户刷新提供的覆盖物添加右键菜单轨迹绘制开发前的准备 账号注册 百度地图开发者平台点此访问 登录注册后点击右上角的控…

【Chapter8】文件系统,计算机操作系统教程,第四版,左万利,王英

文章目录 [toc]一、文件与文件系统1.1 文件1.2 文件系统 二、文件的访问方式2.1 顺序访问2.2 随机访问 三、文件的组织3.1 文件的逻辑组织3.2 文件的物理组织3.2.1 顺序结构3.2.2 链接结构3.2.3 索引结构3.2.4 Hash 结构3.2.5 倒排结构 3.3 UNIX文件物理结构&#xff08;索引链…

深入解读一下 `com.google.android.material.appbar.CollapsingToolbarLayout`

简介 在现代 Android 应用中&#xff0c;提供流畅且美观的用户体验是非常重要的。CollapsingToolbarLayout 是 AndroidX库中 Material Components 的一部分&#xff0c;它提供了一种易于实现的可折叠工具栏效果&#xff0c;常用于提供视觉吸引力的标题栏和动画效果。 本文将详…

printf趣味代码,打印图案

文章目录 1.打印佛祖2.打印猫猫 (闪烁效果) 1.打印佛祖 #include <stdio.h>void budda_bless(){printf("///\n\ // _ooOoo_ //\n\ // o8888888o //\n\ // …

RK3568平台开发系列讲解(调试篇)分析内核调用的利器 ftrace

🚀返回专栏总目录 文章目录 一. 指定 ftrace 跟踪器二、设置要 trace 的函数三、ftrace 的开关四、查看 trace五、trace-cmd 的使用六、trace-cmd 的常用选项6.1、查看可以跟踪的事件6.2、跟踪特定进程的函数调用6.3、函数过滤6.4、限制跟踪深度6.5、追踪特定事件沉淀、分享、…

代码随想录-Day41

46. 携带研究材料&#xff08;第六期模拟笔试&#xff09; 题目描述 小明是一位科学家&#xff0c;他需要参加一场重要的国际科学大会&#xff0c;以展示自己的最新研究成果。他需要带一些研究材料&#xff0c;但是他的行李箱空间有限。这些研究材料包括实验设备、文献资料和实…