Linux c/c++进程间通信(1)

简介: 这篇文章介绍了Linux下C/C++进程间通信的几种方式,包括普通文件、文件映射虚拟内存、管道通信(FIFO),并提供了示例代码和标准输入输出设备的应用。

1. 进程间通信的概念

进程间通信是指不同的进程之间进行信息的传递

    1. 同一主机上的进程通信

            父子进程之间

            非父子进程之间

    2.不同主机上的进程通信(网络通信)

2. 普通文件通信

  1. 父子进程之间可以直接通过文件描述符号(fd)直接进行通信

    **进程之间通过文件通信的模型:**        ![](https://i-blog.csdnimg.cn/blog_migrate/3356fd49c3e7606a216da7b4af5213ba.png)
    

    示例: 父进程写文件,子进程读取文件

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
//父进程写文件    子进程读文件
int main(){
    int fd = open("a.dat",O_RDWR | O_CREAT,0666);
    if(-1 == fd){
        printf("文件创建失败:%m\n"),exit(-1);
    }else{
        printf("文件创建成功!\n");
    }
    close(fd);

    if(fork()){
        //父进程写文件
        int n = 0; 
        while(1){
            int fd = open("a.dat",O_WRONLY);
            if(-1 == fd)
                printf("父: 文件打开失败:%m\n"),exit(-1);
            else
                printf("父: 文件打开成功!\n");
            write(fd,&n,4);
            close(fd);
            sleep(1);
            n+=2;
        }
    }else{
        //子进程读文件
        int m;
        while(1){
            int fd = open("a.dat",O_RDONLY);
            if(-1 == fd)
                printf("子: 文件打开失败:%m\n"),exit(-1);
            else
                printf("子: 文件打开成功!\n");
            sleep(1);
            read(fd,&m,4);
            printf(">> %d\n",m);
            close(fd);
        }
    }

    return 0;
}

运行结果:

  1. 非父子进程之间就只能通过具体的文件来进行通信

3. 文件映射虚拟内存通信

经常用于父子进程之间的通信

#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <unistd.h>
#include <sys/mman.h>
//子进程控制父进程循环的暂停与否(通过文件映射虚拟内存的方式实现)
int main(){
    //1. 打开文件
    int fd = open("mmap.dat",O_CREAT|O_RDWR,0666);
    if(-1 == fd) printf("文件创建失败:%m\n"),exit(-1);
    printf("文件创建成功!\n");
    //2. 修改文件大小
    ftruncate(fd,4);
    //3. 关联映射
    int* p = (int*)mmap(NULL,4,PROT_READ|PROT_WRITE,MAP_SHARED,fd,0);
    if((int*)-1 == p) printf("mmap fail:%m\n"),exit(-1),close(fd);
    printf("mmap sucess!\n");
    *p = 0;
    //4. 使用映射
    if(fork()){
        //父进程
        int n = 0;
        while(1){
            while(*p);
            printf("父进程 >> *p == %d\n",*p);
            printf("父进程 >> %d\n",++n);
            sleep(1);
        }
    }else{
        //子进程
        char ch;
        int n = 0;
        while(1){
            read(0,&ch,1);
            if(ch == ' ')
                n++;

            if(n%2 == 1)
                *p = 1;   //暂停
            else
                *p = 0;   //继续
            printf("子进程: n: %d  *p (%d) 空格被按下!\n",n,*p);
        }
    }
    //5.卸载映射
    munmap(p,4);
    //6.关闭文件
    close(fd);

    return 0;
}

运行结果:

4. 管道通信(first in first out FIFO)

4.1 父子进程之间

    父子进程之间通过**匿名管道**进行通信

    使用**匿名管道**进行通信的步骤:

            1. 创建文件描述符号   **int  fds\[2\];**

            2. 将文件描述符号(fd)变成管道  **pipe**

            3. 使用管道

            4. 关闭文件描述符(**fd**)

示例:

    使用匿名管道  父进程循环等待用户输入并写入管道  子进程循环读取管道内容并输出
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
//使用匿名管道  父进程循环等待用户输入并写入管道  子进程循环读取管道内容并输出
int main(){
//1. 创建文件描述符号   int  fds[2];   fds[0] 读    fds[1] 写
    int fds[2]; 
//2. 将文件描述符号(fd)变成管道  pipe
    int ret = pipe(fds);
    if(-1 == ret) printf("创建管道失败:%m\n"),exit(-1);
    printf("创建管道成功!\n");
//3. 使用管道
    if(fork()){
        char wbuff[1024] = {0};
        while(1){
            scanf("%s",wbuff);
            write(fds[1],wbuff,strlen(wbuff));
            sleep(1);
        }
    }else{
        char rbuff[1024] = {0};
        while(1){
            sleep(1);
            int r = read(fds[0],rbuff,1023);
            if(r > 0){
                rbuff[r]=0;
                printf(">> %s\n",rbuff);
            }
        }
    }
//4. 关闭文件描述符(fd)
    close(fds[0]);
    close(fds[1]);

    return 0;
}

运行结果:

4.2 非父子进程之间

    非父子进程之间通过**有名管道**进行通信

    使用**有名管道**进行通信的步骤:

                  文件A                                                                    文件B

    1\. 创建管道文件 mkfifo

    2. 打开管道文件                                                           2. 打开管道文件

    3. 向管道文件写入数据                                                3\. 从管道文件读取数据

    4. 关闭管道                                                                  4. 关闭管道

    5. 删除管道文件  unlink

文件A:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
//文件A  数据写入端
int main(){
    //1.创建管道文件
    int ret = mkfifo("test.pipe",0666);
    if(-1 == ret) printf("创建管道文件失败:%m\n"),exit(-1);
    printf("创建管道文件成功!\n");

    //2.打开管道文件
    int fd = open("test.pipe",O_WRONLY,0666);
    if(-1 == fd) printf("打开管道文件失败:%m\n"),unlink("test.pipe"),exit(-1);
    printf("打开管道文件成功!\n");

    //3.向管道文件中写入数据
    int n = 0;
    char buff[1024] = {0};
    while(1){
        sprintf(buff,"Linux,学了忘得好快!%d",++n);
        write(fd,buff,strlen(buff));
        sleep(1);
    }

    //4.关闭管道
    close(fd);

    //5.删除管道文件
    unlink("test.pipe");

    return 0;
}

文件B:

#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <fcntl.h>
//文件B  数据读取端
int main(){
    //2. 打开管道文件
    int fd = open("test.pipe",O_RDONLY,0666);
    if(-1 == fd) printf("文件打开失败:%m\n"),unlink("test.pipe"),exit(-1);
    printf("文件打开成功!\n");
    //3. 从管道文件中读取数据
    int r = 0;
    char buff[1024]={0};
    while(1){
        r = read(fd,buff,1023);
        if(r > 0){
            buff[r] = 0;
            printf(">> %s\n",buff);
        }
    }

    //4. 关闭管道
    close(fd);

    return 0;
}

运行结果:

注:

    1. 在**共享文件夹**之中**不能创建管道文件** ![](https://i-blog.csdnimg.cn/blog_migrate/c2d9c0472af45e4e0e10ea6d43ffd12f.png)

    2. 打开管道文件的时候只有**两边一起打开**才会返回,只有一边打开会阻塞

    3. **先关闭读取端,会导致写入端进程结束**        ![](https://i-blog.csdnimg.cn/blog_migrate/6744dcc8317558c3a7833c15baf5d722.png)

    4.**先关闭写入端,读取端进程进入阻塞状态**                ![](https://i-blog.csdnimg.cn/blog_migrate/951ed8af615f844fed1f83c16bdbf2cb.png)

5. 关于标准输入设备(0)与标准输出设备(1)的使用

#include <stdio.h>
#include <stdlib.h>
#include <fcntl.h>
#include <unistd.h>

int main(){
    char buff[1024];
    int r;
    while(1){
        //使用标准输入设备输入
        //输入,向内存中输入,等于从文件读取到内存因此为read
        r = read(0,buff,1023);
        if(r > 0){
            buff[r] = 0;
            //使用标准输出设备输出
            //输出,从内存中输出,等于从内存写到文件因此为write
            write(1,buff,r);
            printf("\n");
        }
    }

    return 0;
}

运行结果:

相关文章
|
27天前
|
弹性计算 人工智能 架构师
阿里云携手Altair共拓云上工业仿真新机遇
2024年9月12日,「2024 Altair 技术大会杭州站」成功召开,阿里云弹性计算产品运营与生态负责人何川,与Altair中国技术总监赵阳在会上联合发布了最新的“云上CAE一体机”。
阿里云携手Altair共拓云上工业仿真新机遇
|
3天前
|
人工智能 Rust Java
10月更文挑战赛火热启动,坚持热爱坚持创作!
开发者社区10月更文挑战,寻找热爱技术内容创作的你,欢迎来创作!
372 16
|
19天前
|
存储 关系型数据库 分布式数据库
GraphRAG:基于PolarDB+通义千问+LangChain的知识图谱+大模型最佳实践
本文介绍了如何使用PolarDB、通义千问和LangChain搭建GraphRAG系统,结合知识图谱和向量检索提升问答质量。通过实例展示了单独使用向量检索和图检索的局限性,并通过图+向量联合搜索增强了问答准确性。PolarDB支持AGE图引擎和pgvector插件,实现图数据和向量数据的统一存储与检索,提升了RAG系统的性能和效果。
|
6天前
|
JSON 自然语言处理 数据管理
阿里云百炼产品月刊【2024年9月】
阿里云百炼产品月刊【2024年9月】,涵盖本月产品和功能发布、活动,应用实践等内容,帮助您快速了解阿里云百炼产品的最新动态。
阿里云百炼产品月刊【2024年9月】
|
21天前
|
人工智能 IDE 程序员
期盼已久!通义灵码 AI 程序员开启邀测,全流程开发仅用几分钟
在云栖大会上,阿里云云原生应用平台负责人丁宇宣布,「通义灵码」完成全面升级,并正式发布 AI 程序员。
|
23天前
|
机器学习/深度学习 算法 大数据
【BetterBench博士】2024 “华为杯”第二十一届中国研究生数学建模竞赛 选题分析
2024“华为杯”数学建模竞赛,对ABCDEF每个题进行详细的分析,涵盖风电场功率优化、WLAN网络吞吐量、磁性元件损耗建模、地理环境问题、高速公路应急车道启用和X射线脉冲星建模等多领域问题,解析了问题类型、专业和技能的需要。
2594 22
【BetterBench博士】2024 “华为杯”第二十一届中国研究生数学建模竞赛 选题分析
|
5天前
|
存储 人工智能 搜索推荐
数据治理,是时候打破刻板印象了
瓴羊智能数据建设与治理产品Datapin全面升级,可演进扩展的数据架构体系为企业数据治理预留发展空间,推出敏捷版用以解决企业数据量不大但需构建数据的场景问题,基于大模型打造的DataAgent更是为企业用好数据资产提供了便利。
182 2
|
3天前
|
编译器 C#
C#多态概述:通过继承实现的不同对象调用相同的方法,表现出不同的行为
C#多态概述:通过继承实现的不同对象调用相同的方法,表现出不同的行为
105 65
|
7天前
|
Linux 虚拟化 开发者
一键将CentOs的yum源更换为国内阿里yum源
一键将CentOs的yum源更换为国内阿里yum源
332 2
|
23天前
|
机器学习/深度学习 算法 数据可视化
【BetterBench博士】2024年中国研究生数学建模竞赛 C题:数据驱动下磁性元件的磁芯损耗建模 问题分析、数学模型、python 代码
2024年中国研究生数学建模竞赛C题聚焦磁性元件磁芯损耗建模。题目背景介绍了电能变换技术的发展与应用,强调磁性元件在功率变换器中的重要性。磁芯损耗受多种因素影响,现有模型难以精确预测。题目要求通过数据分析建立高精度磁芯损耗模型。具体任务包括励磁波形分类、修正斯坦麦茨方程、分析影响因素、构建预测模型及优化设计条件。涉及数据预处理、特征提取、机器学习及优化算法等技术。适合电气、材料、计算机等多个专业学生参与。
1580 17
【BetterBench博士】2024年中国研究生数学建模竞赛 C题:数据驱动下磁性元件的磁芯损耗建模 问题分析、数学模型、python 代码