【并发程序设计】11.进程间通信

news2025/1/19 3:09:58

11.进程间通信

(IPC,InterProcess Communication)进程和进程之间交换信息。

  • 常用通信方式
    • 无名管道(pipe)
    • 有名管道 (fifo)
    • 信号(signal)
    • 共享内存(mmap)
    • 套接字(socket)
  • 过时的IPC通信方式
    • System V IPC
    • 共享内存(share memory)
    • 消息队列(message queue)
    • 信号灯集(semaphore set)

无名管道

无名管道(也称为匿名管道)是UNIX和类UNIX系统下进程间通信的一种机制:

数据读取完后自动删除不会留存

  1. 半双工通信:数据只能在管道的一个方向上流动,具有固定的读端和写端。这意味着在一个时间点上,数据只能在一个方向上被传输,无法同时进行双向通信。
  2. 亲缘关系进程间通信:无名管道通常用于具有亲缘关系的进程之间,如父子进程或兄弟进程之间的通信。一个进程创建管道后,可以通过fork()创建子进程,然后父子进程可以通过这个管道进行数据交换。
  3. 特殊文件形式:无名管道可以看作是一种特殊的文件,对于它的读写可以使用普通的readwrite等函数。但它并不属于任何文件系统,不支持如lseek()操作,且只存在于内存中。

读写特性

  • 读管道:
    • 管道中有数据
      • read返回实际读到的字节数。
    • 管道中无数据:
      • 管道写端被全部关闭,read返回0 (好像读到文件结尾)
      • 写端没有全部被关闭,read阻塞等待(不久的将来可能有数据递达,此时会让出cpu
  • 写管道:
    • 管道读端全部被关闭
      • 进程异常终止(也可使用捕捉SIGPIPE信号,使进程不终止)
    • 管道读端没有全部关闭
      • 管道已满,write阻塞。(管道大小64K)
      • 管道未满,write将数据写入,并返回实际写入的字节数。

pipe函数

  1. 原型

    #include <unistd.h>
    int pipe(int pipefd[2]);
    
  2. 功能pipe函数创建一个管道,用于在两个文件描述符之间进行单向数据传输。一个文件描述符用于读操作,另一个用于写操作。

  3. 参数

    • int pipefd[2]: 一个包含两个整数的数组。
      • pipefd[0]是管道的读端,

      • pipefd[1]是管道的写端。

  4. 返回值

    • 成功时,返回0,并在pipefd中设置两个文件描述符。
    • 失败时,返回-1,并设置errno以指示错误类型。
  5. 注意

    • 只能用于亲缘关系的进程间通信(父子进程,兄弟进程)
    • 管道通信是单工的,一端读,一端写(程序实现设计好)。
    • 数据自己读不能自己写
    • 管道可以用于大于2个进程共享

示例

:一个父进程写,三个子进程读

#include <stdio.h>
#include <unistd.h>
#include <string.h>
int main(){
	int pfd1[2];//文件描述符
	int re,i;
	char buf1[20]={0};
	char buf2[20]={0};
	pid_t pid;
    
    //创建无名管道
	re = pipe(pfd1);
	if(re<0)
	{
		perror("pipe");
		return 0;
	}
    
    //创建三个子进程
	for(i=0; i<3; i++)
	{
		pid = fork();//创建子进程
		if(pid<0)
		{
			perror("fork");
			return 0;
		}
		else if(pid == 0)//如果是子进程就退出循环
			break;
	}

	if(i==3)//父进程
	{
		//close(pfd1[0]);
		while(1)
		{
			strcpy(buf1,"hello world");
			write(pfd1[1],buf1,strlen(buf1));//写入管道

			sleep(1);
		}
	}
	else if(i==0)//子进程0
	{
		//close(pfd1[1]);        
		while(1)
		{
			re=read(pfd1[0],buf2,20);//读取管道数据
			if(re>0)
			{
				printf("0read pfd=%s\n",buf2);
			}    
		}
	}
	else if(i==1)//子进程1
	{
		//close(pfd1[1]);        
		while(1)
		{
			re=read(pfd1[0],buf2,20);//读取管道数据
			if(re>0)
			{
				printf("1read pfd=%s\n",buf2);
			}    
		}
	}
	else if(i==2)//子进程2
	{
		//close(pfd1[1]);        
		while(1)
		{
			re=read(pfd1[0],buf2,20);//读取管道数据
			if(re>0)
			{
				printf("2read pfd=%s\n",buf2);
			}    
		}
	}
}

在这里插入图片描述

有名管道(命名管道)

有名管道,也称为命名管道或FIFO(First In First Out),是一种特殊类型的管道,它允许不相关的进程通过文件系统进行通信。与无名管道相比,有名管道的一个显著特点是提供了一个路径名与之关联,从而可以在任意两个进程之间进行通信。

数据读取完后自动删除不会留存

以下是有名管道的读写特性:

  • 引用计数影响:当所有指向管道写端的文件描述符都关闭时(即写端引用计数为0),如果有进程尝试从读端读取数据,那么管道中剩余的数据被读取后,再次尝试读取会返回0,类似于读到文件末尾的情况。如果写端的引用计数大于0,即仍有文件描述符指向写端且没有写入数据,那么读端在读取完剩余数据后,再次读取会阻塞,直到有新的数据写入。
  • 信号通知:当所有指向管道读端的文件描述符都关闭时(即读端引用计数为0),如果有进程试图向管道中写入数据,该进程会收到一个SIGPIPE信号,通常这会导致进程异常终止。
  • 特殊文件形式:有名管道在文件系统中以特殊文件的形式存在,这意味着它可以通过路径名被访问,并且可以由任何有相应权限的进程使用。这与无名管道只能在亲缘关系的进程间通信不同。
  • 半双工通信:尽管有名管道允许任意进程间通信,但它仍然是一种半双工通信方式,意味着数据只能在一个方向上流动,不能同时进行读写操作。

特点:

  1. 有名管道可以使非亲缘的两个进程互相通信

  2. 通过路径名来操作,在文件系统中可见,但内容存放在内存中

  3. 文件IO来操作有名管道

  4. 遵循先进先出规则

  5. 不支持leek操作

  6. 单工读写

mkfifo函数

  1. 原型

    #include <sys/types.h>
    #include <sys/stat.h>
    int mkfifo(const char *pathname, mode_t mode);
    
  2. 功能mkfifo函数创建一个命名管道(FIFO),该管道存在于文件系统中,可以用于不同进程之间的通信。

  3. 参数

    • const char *pathname: 管道的路径名。指定创建的命名管道的路径和文件名。

    • mode_t mode: 指定文件的权限位,如读、写、执行权限。通常使用宏定义,例如 S_IRUSR | S_IWUSR | S_IRGRP | S_IROTH 等。

  4. 返回值

open函数

  1. 原型

    #include <fcntl.h>
    #include <sys/types.h>
    #include <sys/stat.h>
    #include <unistd.h>
    int open(const char *pathname, int flags, ...);
    
  2. 功能open函数用于打开一个文件,如果成功,则返回一个文件描述符,该文件描述符可用于后续的读写操作。

  3. 参数

    • const char *pathname: 文件的路径名。指定要打开的文件路径。
    • int flags: 文件打开模式和标志位。常用的标志包括:
      • O_RDONLY: 只读模式
      • O_WRONLY: 只写模式
      • O_RDWR: 读写模式
      • O_CREAT: 如果文件不存在则创建文件
      • O_NONBLOCK(非阻塞)可以用|与其他模式连接,不加这个选项表示阻塞
      • O_EXCL: 与 O_CREAT 一起使用,如果文件已存在则返回错误
      • O_TRUNC: 如果文件存在且可写,则将其长度截断为0
      • O_APPEND: 每次写操作都会追加到文件的末尾
    • mode_t mode: 可选参数,仅在使用 O_CREAT 标志时需要。指定文件的权限位,例如 S_IRUSR(用户读权限)、S_IWUSR(用户写权限)、S_IRGRP(组读权限)等。
  4. 返回值

    • 成功时,返回文件描述符。
    • 失败时,返回-1,并设置errno以指示错误类型。

注意事项

  1. 程序不能以O_RDWR(读写)模式打开FIFO文件进行读写操作,而其行为也未明确定义,因为如一个管道以读/写方式打开,进程可以读回自己的输出,同时我们通常使用FIFO只是为了单向的数据传递

  2. 第二个参数中的选项O_NONBLOCK,选项O_NONBLOCK表示非阻塞,加上这个选项后,表示open调用是非阻塞的,如果没有这个选项,则表示open调用是阻塞的

  3. **对于以只读方式(O_RDONLY)**打开的FIFO文件,如果open调用是阻塞的(即第二个参数为O_RDONLY),除非有一个进程以写方式打开同一个FIFO,否则它不会返回;如果open调用是非阻塞的的(即第二个参数为O_RDONLY | O_NONBLOCK),则即使没有其他进程以写方式打开同一个FIFO文件,open调用将成功并立即返回。

    对于以只写方式(O_WRONLY)打开的FIFO文件,如果open调用是阻塞的(即第二个参数为O_WRONLY),open调用将被阻塞,直到有一个进程以只读方式打开同一个FIFO文件为止;如果open调用是非阻塞的(即第二个参数为O_WRONLY | O_NONBLOCK),open总会立即返回,但如果没有其他进程以只读方式打开同一个FIFO文件,open调用将返回-1,并且FIFO也不会被打开。

  4. 数据完整性,如果有多个进程写同一个管道,使用O_WRONLY方式打开管道,如果写入的数据长度小于等于PIPE_BUF(4K),那么或者写入全部字节,或者一个字节都不写入,系统就可以确保数据决不会交错在一起。

示例

:两个没有亲缘关系的进程通信

write.c

#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <string.h>
int main()
{
    int re;
    int fd;
    char buf[32];
    
    //创建管道文件
    re = mkfifo("/myfifo",0666);
    if(re<0)
    {
        perror("mkfifo");
        return 0;
    }
    
    //打开管道文件
    fd = open("/myfifo",O_WRONLY|O_NONBLOCK);//只写非阻塞方式打开
    if(fd<0)
    {
        perror("open");
        return 0;
    }
    printf("after open\n");
    
    //写入管道数据
    while(1)
    {
        fgets(buf,32,stdin);//读取键盘输入
        write(fd,buf,strlen(buf));
    }
}

read.c

#include <sys/types.h>
#include <sys/stat.h>
#include <stdio.h>
#include <unistd.h>
#include <stdlib.h>
#include <fcntl.h>
#include <string.h>
int main()
{
    int re;
    int fd;
    char buf[32];
    fd = open("/myfifo",O_RDONLY);//只读阻塞方式打开管道文件
    if(fd<0)
    {
        perror("open");
        return 0;
    }
    printf("after open\n");
    
    //读取管道数据
    while(1)
    {     
        re=read(fd,buf,32);
        if(re>0)
            printf("read fifo=%s\n",buf);        
        else if(re==0)//
            exit(0);
    }
}

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

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

相关文章

平衡二叉树的构建(理论,部分函数代码)

平衡二叉树是二叉排序树的一种特殊情况&#xff0c;平衡二叉树的出现是为了在最坏情况下的时间复杂度仍然是对数级别O(logn)&#xff0c;从而保证了高效的搜索、插入和删除操作。 举个例子&#xff0c;如果有一个数组是&#xff1a;1&#xff0c;2&#xff0c;3。如果只简单的…

数据库多表查询

多表查询&#xff1a; SELECT *FROM stu_table,class WHERE stu_table.c_idclass.c_id; 多表查询——内连接 查询两张表交集部分。 隐式内连接&#xff1a; #查询学生姓名&#xff0c;和班级名称&#xff0c;隐式调用 SELECT stu_table.s_name,class.c_name FROM stu_table…

vs code 中使用SSH 连接远程的Ubuntu系统

如下图&#xff0c;找到对应的位置 在电脑上找到以下位置 打开配置如下&#xff0c;记住&#xff0c;那个root为你的用户名&#xff0c;这个用户名&#xff0c;具体根据你的用户名来设置&#xff0c;对应的密码就是你登录Ubuntu时的密码 Host root192.168.0.64User rootHostNa…

文件跨境传输法律管控越来越严格,企业该如何有效应对?

文件跨境传输已经成为非常普遍的业务需求&#xff0c;企业在世界各地的总分支处、合作伙伴&#xff0c;客户间开展业务时&#xff0c;必须基于数据的跨境流转而展开。 但随着世界各国对数据安全的重视&#xff0c;文件跨境传输也日趋严格&#xff0c;各国在法律法规上均出具了更…

RP2040 SPI DMA驱动ST7735

SPI DMA演示 一、源码 #include <stdio.h> #include <stdlib.h> #include "pico/stdlib.h" #include "pico/binary_info.h" #include "hardware/spi.h" #include "hardware/dma.h" #include "font.h"#define X…

Ownips+Coze海外社媒数据分析实战指南

目录 一、引言二、ISP代理简介三、应用实践——基于Ownips和coze的社媒智能分析助手3.1、Twitter趋势数据采集3.1.1、Twitter趋势数据接口分析3.1.2、Ownips原生住宅ISP选取与配置3.1.3、数据采集 3.2、基于Ownips和Coze的社媒智能助手3.2.1、Ownips数据采集插件集成3.2.2、创建…

Ubuntu配置Git

安装git sudo apt install git 查看是否安装成功 git --version 配置git 用github上注册的用户名和邮箱地址&#xff0c;配置git git config --global user.name "username" git config --global user.email "usernameemail.com" 重启ubuntu查看…

首搭第五代DM技术,秦L DM-i正式上市,仅售9.98万元起

5月28日&#xff0c;比亚迪王朝重磅新车秦L DM-i在西安震撼上市&#xff0c;首搭第五代DM技术&#xff0c;百公里亏电油耗达到划时代的2.9L&#xff0c;“一箱油”满油满电综合续航达2100公里&#xff0c;引领中级&#xff0c;创下了百公里油耗的历史新低&#xff0c;开创油耗2…

机顶盒也可以跑pcdn--上机指南(贰)

机顶盒能跑PCDN&#xff0c;以下是相关上机指南操作步骤&#xff1a; 1.申请PCDN服务&#xff1a;登录PCDN控制台&#xff0c;申请开通PCDN服务。 2.后台开通PCDN服务&#xff1a;工作人员与用户沟通业务详细需求&#xff0c;用户确定使用PCDN&#xff0c;后台为用户开通PCDN…

VolWeb:集中式增强型数字取证内存分析平台

关于VolWeb VolWeb是一款最新开发的集中式增强型数字取证内存分析平台&#xff0c;该平台基于Volatility 3框架实现其功能&#xff0c;该工具旨在辅助广大研究人员执行安全分析和事件应急响应等任务。 VolWeb可以提供集中式、可视化的增强型网络应用程序&#xff0c;并提高安全…

Java基础:类的详细说明

Java是一门面向对象的编程语言&#xff0c;所谓的面向对象&#xff0c;简单的说&#xff0c;就是在软件开发过程中&#xff0c;用各种各样的对象实现所需功能。 对象就好像是现实世界中不计其数的物体&#xff0c;根据物体的性质可以将其进行分门别类&#xff1a;石头、锤子、…

深入分析 Android Activity (十一)

文章目录 深入分析 Android Activity (十一)1. Activity 的内存管理和优化1.1 内存泄漏的常见原因1.2 避免内存泄漏的方法1.3 内存泄漏检测工具 2. Activity 的配置变更处理2.1 处理配置变更2.2 保存和恢复状态2.3 使用 ViewModel 3. Activity 的测试3.1 单元测试3.2 UI 测试 4…

编译安装Apache httpd服务(LAMP1)

目录 1.初始化设置&#xff0c;将Apache所需软件包传到 /opt 目录下 &#xff08;1&#xff09;关闭防火墙 &#xff08;2&#xff09;上传软件包到/opt目录 2.安装环境依赖包 3.配置软件模块 4.编译及安装 5.优化配置文件路径&#xff0c;并把httpd服务的可执行程序文件…

DDR5芯片系统框图详解

DDR5 SDRAM(双倍数据率五代同步动态随机存取存储器)的功能框图详细展现了其内部结构和各个关键模块,这些模块协同工作以实现高速数据传输和存储管理。以下是主要组成部分的详细介绍: Controller Logic (控制器逻辑)这是DDR5内存系统的大脑,负责接收来自CPU或SoC的指令,解…

怎么从视频中截取图片?这3个视频截图方法超清晰

怎么从视频中截取图片&#xff1f;从视频中截取图片确实是一个不可或缺的技能&#xff0c;特别是在我们想要留住视频中的某个动人瞬间、重要细节或是用于制作海报、封面等场合时。无论是专业的视频编辑人员&#xff0c;还是普通用户&#xff0c;掌握这技巧都能让视频内容得到更…

光缆监测主要功能值得一看

网络资源管理&#xff1a; 设备管理&#xff1a; 测试告警处理&#xff1a; 百度地图&#xff1a; 报表打印&#xff1a; 路由测试数据、路由段数据、路由故障报警统计、当前故障通知情况、路由测试数据、路由段信息。 手机客户端&#xff1a; 通过手机可以访问系统&#x…

为WPF的Grid添加网格边框线

在WPF中使用Grid绘制表格的时候&#xff0c;如果元素较多、排列复杂的话&#xff0c;界面会看起来很糟糕&#xff0c;没有层次&#xff0c;这时用网格或边框线分割各元素&#xff08;标签或单元格&#xff09;将会是页面看起来整齐有条理。 默认没有边框线的如下图所示&#xf…

FuTalk设计周刊-Vol.049

#AI漫谈 热点捕手 1.Gemini Pro1.5及其百万上下文功能现已向所有人开放 Gemini Pro1.5加入视频模态的长上下文功能&#xff0c;AI可以处理更复杂的视频内容。 链接https://aistudio.google.com/app/prompts/new_chat?reftop.aibase.com 2.Figma 2024 Config 大会 6月26-27日…

Java - 当年很流行,现在已经淘汰的 Java 技术,请不要在继续学了!!!

最近这段时间收到了一些读者的私信&#xff0c;问我某个技术要不要学&#xff0c;还有一些在国外的同学竟然对 Java 图形化很感兴趣&#xff0c;还想找这方面的工作。 比较忙&#xff0c;一直没抽出时间去回答这类问题&#xff0c;刚好看到我关注的一位大佬回答过&#xff0c;这…

着色器技术在AI去衣中的魔法般的作用

引言&#xff1a; 在数字图像处理的世界中&#xff0c;AI去衣技术正逐步成为研究的前沿。它利用人工智能的强大能力&#xff0c;实现对图像中衣物的智能识别与处理。在这一过程中&#xff0c;着色器&#xff08;Shader&#xff09;技术扮演了至关重要的角色。本文将深入探讨着色…