Linux 无名管道

无名管道(unnamed pipe)是一种进程间通信的方式,通常用于父子进程之间的通信。下面是使用无名管道的基本步骤:

pipe 调用的返回值如下:

  • 成功时pipe 调用成功时返回 0。
  • 失败时 :如果 pipe 调用失败,返回 -1,并设置 errno 以指示错误原因。errno 可能被设置为以下几种值之一:
    • EBADF:文件描述符不正确。
    • EACCES:没有权限创建管道。
    • EMFILE:进程已经达到了它能够打开的文件描述符的最大数量。
    • ENFILE:系统已经达到了能够打开的文件的最大数量。
    • ENOMEM:没有足够的内核内存资源来创建管道。
  1. 创建一个无名管道:使用 pipe() 函数创建一个无名管道,它会返回两个文件描述符,fd0 用于读取数据,fd1 用于写入数据。
c 复制代码
int fd[2];
pipe(fd);
  1. 创建子进程:使用 fork() 函数创建一个子进程,子进程会继承父进程的文件描述符。
c 复制代码
pid_t pid = fork();
  1. 在父进程中写入数据:如果当前是父进程,则可以使用 fd1 来写入数据到无名管道。
c 复制代码
if (pid > 0) {
    close(fd[0]);  // 关闭读取端
    write(fd[1], data, sizeof(data));
    close(fd[1]);  // 关闭写入端
}
  1. 在子进程中读取数据:如果当前是子进程,则可以使用 fd0 来读取从父进程写入的数据。
c 复制代码
if (pid == 0) {
    close(fd[1]);  // 关闭写入端
    read(fd[0], buffer, sizeof(buffer));
    close(fd[0]);  // 关闭读取端
}

注意事项:

  • 父进程和子进程在使用无名管道时需要关闭不使用的一端,以避免造成不必要的阻塞。
  • 无名管道是一个单向通道,数据只能单向流动,需要双向通信时可以创建两个无名管道。
  • 无名管道只能在具有共同祖先的进程之间使用。
  • 无名管道的大小是有限的,一般为 65536 字节,超过该大小的数据会被截断。

这是一个简单的示例,请根据实际情况进行相应的修改和扩展。

cpp 复制代码
//无名管道(pipe)
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <stdlib.h>
#include <sys/wait.h>
#include <string.h>
int main(void){
        int fd[2];//两个管道
        char buf[100];//用于存储发送和接收的内容
        if(pipe(fd)<0){//管道失败的话
                perror( "pipe");
                exit(1);
        }
        printf("无名管道开始\n");
        pid_t fork_ret=fork();
        if(fork_ret<0){
                perror("fork()");
                return -1; 
        }else if(fork_ret==0){
                while(1){
                        bzero(buf,sizeof(buf));
                        printf("\n大好儿:PID:%d  PPID:%d\n\n",getpid(),getppid());
                        read(fd[0],buf,sizeof(buf));
                        printf("呀 是 爹!!发来的:%s \n",buf);
                        if( strncmp(buf,"quit",4) == 0){ 
                                break;
                        }
                }
        }else{
                while(1){
                        bzero(buf,sizeof(buf));
                        printf("爹:PID:%d  PPID:%d\n",getpid(),getppid());
                        fgets(buf,sizeof(buf),stdin);
                        printf("向大好儿发送:%s \n\n",buf);
                        write(fd[1],buf,strlen(buf));
                        if( strncmp(buf,"quit",4) == 0){ 
                                break;
                        }               }       }
        return 0;
}
:se
相关推荐
来自于狂人几秒前
第5章 记忆管理——让Agent记住事情
人工智能·算法·语言模型·自然语言处理
江湖有缘2 分钟前
Docker部署HamsterBase Tasks任务管理工具
运维·docker·容器
CHHH_HHH4 分钟前
【C++】哈希表原理与实战:从冲突解决到性能优化
开发语言·数据结构·c++·学习·算法·哈希算法·散列表
sali-tec12 分钟前
C# 基于OpenCv的视觉工作流-章84-包胶有无检测
图像处理·人工智能·opencv·算法·计算机视觉
Irissgwe19 分钟前
数据结构-排序
数据结构·算法·排序算法
huangdong_20 分钟前
1688商品图片批量下载与SKU图自动分类技术完整实现方案
运维·服务器
小O的算法实验室23 分钟前
2025年IEEE TITS,基于动态聚类粒子群算法的无人机任务分配与路径规划
算法
Tairitsu_H28 分钟前
[LC优选算法#5] 分治:快排 | 颜色分类 | 排序数组 | 第K大元素
c++·算法·leetcode·排序算法·快速排序
yyuuuzz28 分钟前
独立站运营的几个技术层面常见问题
大数据·运维·服务器·网络·数据库·aws
utf8mb4安全女神28 分钟前
MySQL8.0.43的下载安装【二进制安装】【shell脚本】【环境准备】【my.cnf配置】【修改密码】
linux·服务器·网络