美文网首页
13 进程间通讯: 管道

13 进程间通讯: 管道

作者: helinyu | 来源:发表于2017-01-15 21:59 被阅读87次

管道###

【11 章中,我们看到一种在两个进程间发送消息的非常简单的方法: 使用信号;我们创建通知事件,通过它引起响应,但传送的信息只限于一个信号值】
管道:通过它进程之间可以交换更有用的数据。
本章后:我们将学习到的知识将CD数据库应用程序重新实现为一个非常简单的客户/服务器应用程序。

介绍大概:
(1)管道的定义
(2)进程管道
(3)管道调用
(4)父进程和子进程
(5)命名管道:FIFO
(6)客户/服务器架构

13.1 什么是管道?
当从一个进程连接数据流到另一个进程时,我们使用术语:管道(pipe)。
(我们通常是把一个进程的输出通过管道连接到另一个进程的输入)
大多数Linux的用于应该早已对将shell命令连接在一起的概念很熟悉了,这实际上就是把一个进程的输出直接传递给另外一个进程的输入。(shell命令连接在一起???“|”)
eg: cmd1 | cmd2
对于shell命令俩说,命令的连接是通过管道字符来完成的。【管道字符:“|”】
shell负责安排两个命令的标准输入和标准输出
(1)cmd1 的标准输入来自终端的键盘
(2)cmd1 的标准输出传递给cmd2 ,作为它的标准输入;
(3)cmd2的标准输出连接到终端屏幕。
. shell 所做的工作实际上就是对标准输入和标准输出流进行了重新连接,是数据连接从键盘输入通过两个命令最终输出到屏幕上;

管道图示

本章实现包含了 : 如何在程序中获得这样的效果,怎样用管道将多个进程连接起来,从而实现一个简单的客户/服务器系统。

13.2 进程管道###

两个最简单的在两个程序之间传递数据的方法就是使用popen 和pclose 函数了。
头文件: stdio.h
FILE *popen(const char *command, const cahr *open_mode);
int pclose(FILE *stream_to_close);
(1)popen 函数
popen 函数允许一个程序将另外一个程序作为新进程启动,并可以传递数据给它或者通过它接收数据。
command 字符串是要运行的程序名和相应的参数。
open_mode 必须是“r”或“w”;
如果open_mode 是“r”,被调用程序的输出就可以被 调用程序使用,调用程序利用popen函数返回的FILE * 文件流指针,就可以通过常用的stdio库函数(如:fread)来读取被调用程序的输出;
如果open_mode 是“w”,调用程序就可以用fwrite 调用向被调用程序发送数据,而被调通常用程序可以在自己的标准输入上读取这些数据。被调用的程序通常不会意识到自己正在从另外一个进程读取数据,它只是在标准输入留上读取数据,然后做出相应的操作。
每个popen调用都必须指定“r”或“w”,在popen函数的标准实现中不支持任何其他选项。这意味着我们不能调用另外一个程序并同时对它进行读写操作(只可以选择一个)。popen函数在失败时返回一个空指针。如果想通过管道实现双向通讯,最普通的解决方法是使用两个管道,每个管道负责一个方向的数据流。

(2)pclose函数####

用popen启动的进程结束时,我们就可以用pclose函数关闭与之关联的文件流。##

pclose 调用值在popen启动的进程结束后才返回。如果调用pclose时它仍然在云心,pclose调用将等待该进程的结束。
pclose 调用的返回值通常是它所关闭的文件流所在进程的退出码。如果调用进程在调用pclose之前执行了一个wait语句,被调用进程的退出状态就会丢失,因为被调用进程已经结束。此时,pclose将返回-1并设置errno为ECHILD。

实验:读取外部程序的输出###

打印系统信息命令
下面的示例是在程序中用popen访问uname命令给出的信息。
大概过程 : 完成程序的初始化工作后,打开一个连接到uname命令的管道,把管道设置为可读方式并让read_fp 指向命令的输出。最后,关闭read_fp指向的管道。
代码以及运行的结果
实验解析: 这个程序用popen 调用启动带有-a选项的uname命令。然后用返回的文件流读取最多BUFSIZge个字符(这个常量是stdio.h 中用#define 语句定义的)的数据,并将它们打印出来显示在屏幕上。因为我们是子啊程序内部捕获uname命令的输出,所以可以处理它。
实例代码

13.3将输出送往popen###

上面一个例子是捕获外部程序输出的例子;下面是将一个输出发送到外部程序。

示例代码
这里是使用带有参数“w”的popen 启动od -c 命令,这样就可以像该命令发送数据了。然后它给od -c命令发送一个字符串,该命令接受并且处理它,最后把处理结果打印到自己的标准输出上。
相当于:
等价命令
代码

13.3.1 传递更多的数据####

目前所使员工的数据通过一个fread 或fwrite 调用来发送或接受; 有时,我们可能希望以块的方式发送数据,或者我们根本不知道输出数据的长度。为了避免顶一个分厂大的缓冲区,我们可以用多个fread或fwrite调用数据分为几部分处理。


代码需要调整,没有获取所说的结果

这个程序调用了popen函数使用“r”参数,和popen1.c 程序的做法一样。这一次我们是连续从文件流中读取数据,知道没有数据读取为止。【注意:虽然ps命令的执行要花费一些时间,单linux会安排好进程间的调度,让两个程序可以运行时继续运行】如果进程popen3没有数据可读,它将被挂起知道数据到达。如果写进程ps产生的输出超过了可用缓冲区的长度,它也会被挂起知道读进程读取一些数据。

13.3.2如何实现popen###

请求popen调用运行一个程序时,它首先启动shell,即系统中的sh命令,然后将command字符串作为一个参数传递给它。这有两个效果,一个好一个不太好。
在Linux(Unix) 中,所有参数扩展都是由于shell来完成的。所以,启动程序之前先启动shell来分析命令字符串,就可以使各种shell扩展(eg:c所指的是哪些文件)。在程序启动之前就会全部完成。这个功能是非常有用的,它允许我们通过popen启动非常复杂的shell命令。而其他一些创建进程的函数(eg:execl)调用起来就复杂很多,因为调用进程必须自己去完成shell扩展。
使用shell不太好的影响是:针对每个popen调用,不仅要启动一个呗请求的程序,还要启动一个shell,即每个popen调用将多启动两个进程。从节省系统资源的角度来看,popen函数的调用成本略搞,而且对目标命令的调用比正常方式要慢一些。
下面的命令: cat popen
.c | wc -l

代码
shell在启动后将popen*.c 扩展为一个文件列表,列表中的文件名都是以popen开头,以.c 结尾,shell还处理了管道符(|)并将cat命令的输出传递给wc命令。我在一个popen调用中启动了shell、cat 程序和wc程序,并进行了一次输出重定向。而调用这些命令的程序值看到最终的输出结果。

总结上面:
上面主要是关于什么是管道,就是进程之间的一种输入和输出的之间的信息交流。
进程管道的使用以及popen这个函数的使用。

相关文章

  • 13 进程间通讯: 管道

    管道### 【11 章中,我们看到一种在两个进程间发送消息的非常简单的方法: 使用信号;我们创建通知事件,通过它引...

  • 系统编程-------进程间通讯

    进程间通讯 pipe, 亲属间进程通讯, 参数: pipedfd :用于接收pipe函数创建的管道文件的读写...

  • 操作系统知识梳理

    (一)进程间的通信方式 管道中还有命名管道和非命名管道之分,非命名管道只能用于父子进程通讯,命名管道可用于非父子进...

  • STHO

    管道实现进程间通讯https://blog.csdn.net/skyroben/article/details/7...

  • 进程间通讯

    一、进程间通讯的方式进程间通讯的方式有很多,常用的有共享内存(内存映射文件、共享内存DLL、剪切板等)、命名管道和...

  • python进程间的通讯之管道通讯

    from mumultiprocessing import Process,Pipe import os def ...

  • Android Binder

    Binder整理 1、IPC: 进程间通讯或者夸进程通信windows ipc方式:剪贴板、管道、邮槽等等linu...

  • Android-handler 分析

    用于同进程的不同线程进行通讯,不能用于进程间通讯,Handler机制中管道作用就是当一个线程A准备好Message...

  • 进程通信的五种方式

    1.管道:速度慢,容量有限,只有父子进程能通讯 2.FIFO:任何进程间都能通讯,但速度慢 3.消息队列:容量受到...

  • Linux内核及其功能

    管理进程:内核负责创建和销毁进程, 并处理它们与外部世界的联系(输入和输出),不同进程间通讯(通过信号,管道,或者...

网友评论

      本文标题:13 进程间通讯: 管道

      本文链接:https://www.haomeiwen.com/subject/gkdubttx.html