进程通信——匿名管道

有的时候在程序的开发过程中,两个进程之间会有数据的交互。信号的机制能够实现进程通信。它通过 “中断--响应” 的异步模式来工作。但作为通信来讲,信号还是远远不够的。因为它不能够携带任何其他的信息。只利用信号机制来实现进程通信显得捉襟见肘,并且信号的优势并不此。所以必须开发新的进程间通信的方法。本文所学习的 "匿名管道" 便是其中的一种最简单的方法。

基本概念

在讲管道的基本概念之前,首先提一个问题。如果让你来设计进程通信的方式,你会怎么做?最简单的方式莫过于两个进程打开同一个文件,这样就可以通过不断的读写操作来实现信息的交互。管道的概念与原理与上面的方法类似但不完全一样。下面来看下创建管道的函数 pipe

#include <unistd.h>
int pipe(int fd[2]);
                    返回值:若成功,返回0,若出错,返回-1

该函数返回两个文件描述符 fd[0]fd[1]。规定:fd[0]用于读操作,fd[1]用于写操作。在创建管道之后,通常的做法是调用 fork 函数,那么子进程也就继承了父进程的fd[0]fd[1]。所以,这样父进程与子进程就通过这两个文件描述符建立了一个通信的 "管道" 。在通信时,父进程与子进程要么一个关闭读端要么一个关闭写端来实现 半双工 方式的数据传递。即数据流只能往一个方向走而不能同时通信。下面的这个图有助于理解 管道 的这种工作方式:

之所以称之为 "匿名" 是因为这种技术只能适用在具有亲缘关系的两个进程中。不是任意的两个进程都可以通过这种方式来实现通信。

简单的例子

下面的这个例子利用管道来实现了,子进程向父进程传递一段信息,父进程接收到并打印出来的简单功能。

int main(void)
{
    int n;
    int fd[2];
    pid_t pid;

    char line[1024];

    if (pipe(fd) < 0)
        return;
    if ((pid = fork()) < 0)
        return;
    else if (pid > 0)
    {
        close(fd[0]);
        write(fd[1], "hello world\n", 12);
    }
    else
    {
        close(fd[1]);
        n = read(fd[0], line, 1024);
        printf("the recv msg is %s\n", line);
    }
    return(0);

}

注意点

这里记录一下,管道操作中对读端,写端操作的注意点。

  • 写端没有关闭的情况下,执行 read 操作:

    1. 如果管道中没有数据,read 将会阻塞。
    2. 管道的缓冲区也是有大小的,这个值记录在PIPE_BUF中。 read函数传入的字节数、管道中实际的字节数、管道缓冲区的大小这三者之间 会存在大小顺序的关系。read的返回值也跟着三者之间的关系有关。不过只要记住, read返回的是实际读到的字节数,并做好返回值检查就可以在编程中避免错误了。
  • 写端关闭的情形下,执行 read 操作:

    这个时候read不会出错,但是返回值为0。

  • 读端未关闭的情况下,执行 write 操作:

    如果管道的缓冲区已经满了,则write操作阻塞

  • 读端关闭的情况下,执行 write 操作:

    这种情况是危险的,write返回-1, error设为EPIPE, 并产生SIGPIPE信号。

popen pclose函数

在进程控制里面提到过,如果想在程序里面调用 shell 命令或者脚本文件,可以通过fork函数自己实现,或者直接调用 system 函数。现在我们又多了一种方法,这就是popen函数,并且popen函数通过管道机制,可以将执行shell命令的输出结果传递到父进程,或者父进程可以通过管道向shell命令传递参数。这样的实现又极大的方便了我们编程的需求。下面先写出popen pclose函数的声明,再举几个例子来加深一下印象。

#include <stdio.h>
FILE *popen(const char *cmdstring, const char *type);
                            返回值:若成功,返回文件指针;若出错,返回NULL

int pclose(FILE *fp);
                            返回值:若成功,返回cmdstring的终止状态;若出错,返回-1

popen 函数传入的 type 为 "r" 时,表示程序可以从其返回的文件指针中读取执行 cmstring 的结果,它其实将子进程的 stdout 与返回的文件指针连接在一起。如下图所示:

popen 函数传入的 type 为 "w" 时,表示父进程可以通过文件指针来向子进程传递参数,它的理解可以借助于下图:

两个简单的利用 popen 函数的例子:

/* test popen output */
int main()
{
    FILE *fp;
    pid_t pid;
    char *cmd = "ls -al";
    char line[1024];

    if ((fp = popen(cmd, "r")) == NULL)
        return;

    // get and print the data from child process
    fread(line, sizeof(char), 1024, fp);
    printf("%s\n", line);

    pclose(fp);
    return;
}

/* test popen input */
int main()
{
    FILE *fp;
    pid_t pid;
    char *cmd = "cat ";
    char *msg = "hello world\n";

    if ((fp = popen(cmd, "w")) == NULL)
        return;

    // get the params form the parent process
    fwrite(msg, sizeof(char), 12, fp);
    pclose(fp);

    return;
}
时间: 12-19

进程通信——匿名管道的相关文章

Linux进程通信----匿名管道

Linux进程通信中最为简单的方式是匿名管道 匿名管道的创建需要用到pipe函数,pipe函数参数为一个数组 表示的文件描述字.这个数组有两个文件描述字,第一个是用 于读数据的文件描述符第二个是用于写数据的文件描述符. 不能将用于写的文件描述符进行读操作或者进行读的文件描述 符进写操作,这样都会导致错误. 关于匿名管道的几点说明: 1.匿名管道是半双工的,即一个进程只能读,一个进程只能写    要实现全双工,需要两个匿名管道. 2.只能在父子进程或者兄弟进程进行通信. 3.在读的时候关闭写文件描

进程通信——有名管道

上一篇文章中学习了进程间通信的一种简答的方法:匿名管道.但是它只能用于具有亲缘关系的进程之间的通信.而FIFO的通信机制与之相似,却可以在任意两个进程之间通信. FIFO文件操作 创建FIFO类似于创建文件,确实,FIFO可以存在于文件系统中.下面是创建FIFO的函数: #include <sys/stat.h> int mkfifi(const char *pathname, mode_t mode); 返回值:若成功返回0,若出错返回-1 参数说明: pathname: FIFO文件的路径

Linux学习笔记(13)-进程通信|命名管道

匿名管道只能在具有亲属关系的进程间通信,那么如果想要在不具有亲戚关系,想在陌生人之间通信,那又该怎么办呢? 别慌,Linux身为世界上*强大的操作系统,当然提供了这种机制,那便是命名管道-- 所谓命名管道,那便是拥有名字的管道,同时也被称之为FIFO,谈到FIFO,那么做过单片机开发的同学想必是不陌生的. 在很多单片机的项目中,都使用过FIFO,FIFO其实是一种队列,先进先出,这样可以保证读出数据和写入数据的一致性. 使用FIFO文件,便可以在不同的,且不具有亲属关系的进程中进程通信. 创建命

进程通信-无名管道

无名管道:主要是针对进程通信的(自己感觉它有很大的局限性) 特点:它不是一个文件系统,不能按名访问,这也是它和有名管道之间最大的区别.无名管道只是一个系统内存里面的东西. 半双工模式,数据只能流向一个方向(老师给我们举得例子就是水厂的水到居民用水,不可能倒着流对吧). 进程之间通信,但是只能是有亲缘关系的进程才能进行通信?比如父子进程:因为在父子进程中,子进程拷贝父进程的数据段,这让这两个进程在通信的时候就有了相连的关系. 下面我们来看一下:无名管道的创建,写入,读取,关闭(我用的的是Liunx

进程通信之管道

本节学习进程通信的另一种方式:管道.管道是一个进程连接数据流到另一个进程的通道,它通常把一个进程的输出通过管道连接到另一个进程的输入.在shell命令中经常会看到管道的应用,比如我们要列出当前文件下所有命名中有"test"的文件:ls -l | grep test.其中"|"就代表我们在使用管道,它会把"ls -l"的查询结果通过管道,发送给grep,然后执行"grep test"命令后把结构输出到终端.总之一句话:管道会把一

linux进程通信之管道

1.介绍: 1)同一主机: unix进程通信方式:无名管道,有名管道,信号 system v方式:信号量,消息队列,共享内存 2)网络通信:Socket,RPC 2.管道: 无名管道(PIPE):使用一个进程的标准输出作为另一个进程的标准输入建立的一个单向管道,执行完成后消失.主要用于父进程与子进程之间,或者两个兄弟进程之间.采用的是单向 1)创建无名管道:(#include(unistd.h)) extern int pipe(int pipes[2]),pipes[0]完成读操作,pipes

Linux下进程通信之管道

每个进程各自有不同的用户地址空间,任何一个进程的全局变量在另一个进程中都看不到,所以进程之间要交换数据必须通过内核,在内核中开辟一块缓冲区,进程1把数据从用户空间拷到内核缓冲区,进程2再从内核缓冲区把数据读走,内核提供的这种机制称为进程间通信(IPC,InterProcess Communication).如下图所示. 目前进程通信的方式有: 管道 FIFO 消息队列 信号量 共享内存 套接字 管道 管道概念 管道是Linux支持的最初Unix IPC形式之一,具有以下特点: 管道是半双工的,数

linux下的进程通信之管道

概念:管道是由内核管理的一个缓冲区,相当于我们放入内存中的一个纸条.管道的一端连接一个进程的输出.这个进程会向管道中放入信息.管道的另一端连接一个进程的输入,这个进程取出被放入管道的信息. 优点:不需要加锁,基于字节流不需要定义数据结构 缺点:速度慢,容量有限,只能用于父子进程之间,使用场景狭窄 基本原理: 一个缓冲区不需要很大,它被设计成为环形的数据结构,以便管道可以被循环利用.当管道中没有信息的话,从管道中读取的进程会等待,直到另一端的进程放入信息.当管道被放满信息的时候,尝试放入信息的进程

Linux 进程通信之管道

管道是单向的.先进先出的,它把一个进程的输出和另一个进程的输入连接在一起.一个进程(写进程)在管道的尾部写入数据,另一个进程(读进程)从管道的头部读出数据.数据被一个进程读出后,将被从管道中删除,其他读进程将不能再读到这些数据.管道提供了简单的流控制机制,进程试图读空管道时,进程将阻塞.同样,管道已经满时,进程再试图向管道写入数据,进程将阻塞. 管道包括无名管道和有名管道两种,无名管道只能用于父进程和子进程间的通信,而有名管道可以用于同一系统中的任意两个进程间的通信. 无名管道由pipe()函数