对于使用管道的一些理解
看到如下这个连接的帖子,我觉得这是个很好的讨论主题,感觉想写点东西。因此就按照网友andyY和gadfly版主的思路继续写了下去。 http://www.chinaunix.net/forum/viewtopic.php?t=93300 1、首先看看正常的情况。 测试代码如下 [code:1:a6839c86ed] #includes
看到如下这个连接的帖子,我觉得这是个很好的讨论主题,感觉想写点东西。因此就按照网友andyY和gadfly版主的思路继续写了下去。
http://www.chinaunix.net/forum/viewtopic.php?t=93300
1、首先看看正常的情况。
测试代码如下
[code:1:a6839c86ed]
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <errno.h>
#define WR 100000
int main(void)
{
char wrline[WR];
int fd[2];
int res;
pid_t pid;
memset(wrline,'K',WR);
if(pipe(fd) < 0 ) {
(void)fprintf(stderr,"pipe
error:errno[%d],errmsg[%s]\n",errno,strerror(errno));
exit(-1);
}
pid=fork();
if(pid==0)
{
close(fd[1]);
sleep(20);
while(1) {
if( ( res=read(fd[0],wrline,WR-1) ) < 0 ) {
(void)fprintf(stderr,"read
error:errno[%d],errmsg[%s]!\n",errno
,strerror(errno));
break;
}
(void)fprintf(stderr,"child[%d] process read: %d\n",getpid(),res);
}
exit(0);
}
else if( pid > 0 ) {
close(fd[0]);
if( ( res=write(fd[1],wrline,WR-1) ) < 0 ) {
(void)fprintf(stderr,"write
error:errno[%d],errmsg[%s]!\n",errno,str
error(errno));
}
(void)fprintf(stderr,"parent[%d] process write: %d\n",getpid(),res);
pause();
} else {
(void)fprintf(stderr,"fork
error:errno[%d],errmsg[%s]!\n",errno,strerro
r(errno));
exit(-1);
}
exit(0);
}
[/code:1:a6839c86ed]
如上代码安排的属于正常情况,在pipe后,父进程关闭pipe读的一端,子进程关闭写的一端。
然后父进程写10万字节的数据到pipe。只要内核存储器条件满足,可以写足够大的数据量,例
如100万。父进称写成功后调用pause()是有目的的。这一点将在下面的测试中说明。
我们让子进程sleep(20)的目的是为了保证让父进程写管道顺利完成(所谓的顺利是考虑到写管
道时的数据可能超过了内核参数PIPE_BUF规定的管道缓存区的大小,这时候写的动作往往由内核
分为多次完成)。
代码很简单,不作过多介绍,下面我们看看在不同的系统上面的测试结果,遗憾的是,我这里
的Sun OS环境损坏了,只能测试hp,aix,sco上面的情况。
(1)、在hp-11上面测试结果如下:
[code:1:a6839c86ed]
child[1568] process read: 8192
child[1568] process read: 8192
child[1568] process read: 8192
child[1568] process read: 8192
child[1568] process read: 8192
child[1568] process read: 8192
child[1568] process read: 8192
child[1568] process read: 8192
child[1568] process read: 8192
child[1568] process read: 8192
child[1568] process read: 8192
child[1568] process read: 8192
child[1568] process read: 1696
parent[1567] process write: 100000
[/code:1:a6839c86ed]
我们看到在hp-11上,pipe缓存设置为8192.父进程按照pipe缓存的大小分为多次将10万字节
的数据成功写入pipe。
(2)、在aix5上面的测试结果如下:
[code:1:a6839c86ed]
child[48770] process read: 32768
child[48770] process read: 32768
child[48770] process read: 32768
child[48770] process read: 1696
parent[48278] process write: 100000
[/code:1:a6839c86ed]
我们看到aix5的管道缓存大小为32768。这是int的上限值。
(3)、SCO unix 5.05上面的测试结果如下:
[code:1:a6839c86ed]
child[1110] process read: 8192
child[1110] process read: 8192
child[1110] process read: 8192
child[1110] process read: 8192
child[1110] process read: 8192
child[1110] process read: 8192
child[1110] process read: 8192
child[1110] process read: 8192
child[1110] process read: 8192
child[1110] process read: 8192
child[1110] process read: 8192
child[1110] process read: 8192
parent[1109] process write: 99999
child[1110] process read: 1695
[/code:1:a6839c86ed]
我们看到SCO5.05系统提供的管道缓存大小为8192.
总结:
综合以上三个系统来看,我们发现write数据到pipe,并不阻塞,不论这些数据有没有被其它的
进程读走。为了保证write彻底的将所有数据写完,可以让子进程sleep更长的时间。往一个管
道中写数据受到系统PIPE_BUF的限制。
2、写一个读端已经被关闭的管道。
测是代码如下:
[code:1:a6839c86ed]
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <errno.h>
#include <signal.h>
#define WR 100000
void sig_pipe( int signo )
{
(void)fprintf(stderr,"signal[%d] is capture.\n", signo );
}
int main(void)
{
char wrline[WR];
int fd[2];
int res;
pid_t pid;
memset(wrline,'K',sizeof(wrline));
if(pipe(fd) < 0 ) {
(void)fprintf(stderr,"pipe
error:errno[%d],errmsg[%s]\n",errno,strerror(
errno));
exit(-1);
}
signal(SIGPIPE,sig_pipe);
pid=fork();
if(pid==0)
{
close(fd[0]);/* 我们特意关闭管道的读端,也可以用exit(0)让子进程直接退出 */
close(fd[1]);
while(1) {
if( ( res=read(fd[0],wrline,WR-1) ) < 0 ) {
(void)fprintf(stderr,"read
error:errno[%d],errmsg[%s]!\n",errno
,strerror(errno));
break;
}
(void)fprintf(stderr,"child[%d] process read: %d\n",getpid(),res);
}
exit(0);
}
else if( pid > 0 ) {
close(fd[0]);
sleep(5);
if((res=write(fd[1],wrline,WR)) < 0 ) {
(void)fprintf(stderr,"write
error:errno[%d],errmsg[%s]!\n",errno,st
rerror(errno));
}
(void)fprintf(stderr,"parent[%d] process write: %d\n",getpid(),res);
pause();
} else {
(void)fprintf(stderr,"fork
error:errno[%d],errmsg[%s]!\n",errno,strerro
r(errno));
exit(-1);
}
exit(0);
}
[/code:1:a6839c86ed]
如上的代码,我们想办法让管道的读端关闭。一般的可以直接调用close便可,如果子进程调用
exit()也可以,因为我们知道进程退出后其打开的描述字被关闭,管道便是其中的一种,因此
可以达到相同的目的。这段解释也说明了为什么第一个示例代码用pause()二不用exit()的原
因。
我们让父进程sleep(5)的目的时保证在write的时候管道的读端已经被关闭。
signal(SIGPIPE,sig_pipe);语句用来在fork之前设置对SIGPIPE信号处理程序。
信号处理程序sig_pipe很简单,仅仅想标准输出打印一条消息,以便我们能够知道进程中发生
了SIGPIPE信号。
下面看看不同系统的表现形式:
在hp-11、aix5、sco 5.05上面的我们得到相同的输出,如下所示:
[code:1:a6839c86ed]
read error:errno[9],errmsg[Bad file number]!
signal[13] is capture.
write error:errno[32],errmsg[Broken pipe]!
parent[1848] process write: -1
[/code:1:a6839c86ed]
可以看出,当fd[0]被彻底关闭后(彻底是指父子进程都关闭的情况,因为fork后,子进程继承
了父进程的fd[0]和fd[1]。因此彻底的关闭需要父子进程同时close).
子进程从管道读时产生错误Bad file number。表明fd[0]已经不是有效的描述字。需要说明的
是:如果子进程用exit(0)测试这种情况,这里将不会看到read输出信息的。
父进程write时,产生错误Broken
pipe,也就是EPIPE。同时,我们的信号处理程序显示signal[13] is capture.表明确实在向
读端关闭的管道中写时(这里时父进程写)产生了信号_SIGPIPE(也就是SIGPIPE,这往往定义
为一个宏)。
3、读一个写端已经被关闭的管道。
测试代码如下:
[code:1:a6839c86ed]
#include <stdio.h>
#include <unistd.h>
#include <sys/types.h>
#include <errno.h>
#define WR 100000
int main(void)
{
char wrline[WR];
int fd[2];
int res;
pid_t pid;
memset(wrline,'K',sizeof(wrline));
if(pipe(fd) < 0 ) {
(void)fprintf(stderr,"pipe
error:errno[%d],errmsg[%s]\n",errno,strerror(
errno));
exit(-1);
}
pid=fork();
if(pid==0)
{
close(fd[1]);
sleep(5);
while(1) {
if( ( res=read(fd[0],wrline,WR) ) <= 0 ) {
(void)fprintf(stderr,"read
error:errno[%d],errmsg[%s]!\n",errno
,strerror(errno));
(void)fprintf(stderr,"read: %d\n",res);
break;
}
(void)fprintf(stderr,"child[%d] process read: %d\n",getpid(),res);
}
exit(0);
}
else if( pid > 0 ) {
close(fd[0]);
if( ( res=write(fd[1],wrline,WR) ) < 0 ) {
(void)fprintf(stderr,"write
error:errno[%d],errmsg[%s]!\n",errno,st
rerror(errno));
}
(void)fprintf(stderr,"parent[%d] process write: %d\n",getpid(),res);
close(fd[1]);
} else {
(void)fprintf(stderr,"fork
error:errno[%d],errmsg[%s]!\n",errno,strerro
r(errno));
exit(-1);
}
exit(0);
}
[/code:1:a6839c86ed]
如上的代码,我们让子进程sleep(5)的目的是保证让父进程写完数据后关闭管道后子进程在去
读取pipe。
父进程中close(fd[1])可以用exit(0)代替。作用是相同的。
我们在hp-11/aix5/SCO5.05上面看到了相似的输出过程:
[code:1:a6839c86ed]
child[2004] process read: 8192
child[2004] process read: 8192
child[2004] process read: 8192
child[2004] process read: 8192
child[2004] process read: 8192
child[2004] process read: 8192
child[2004] process read: 8192
child[2004] process read: 8192
child[2004] process read: 8192
child[2004] process read: 8192
child[2004] process read: 8192
child[2004] process read: 8192
child[2004] process read: 1696
parent[2003] process write: 100000
read error:errno[0],errmsg[Error 0]!
read: 0
[/code:1:a6839c86ed]
三个系统中不同的仅仅是缓存大小的差别。最后read都返回0。
我们看到,尽管管道的写端已经被关闭,然而在管道读端可用的情况下,内核保证管道中的数
据仍然能够被读取。
当管道中的数据被读取完后,read返回0。返回0并不标识一个错误,这往往有内核给调用者返
回一个文件结束符来标志。posix建议对于这种情况,分为进程存在或不存在分别处理,但是,
在上面的三个系统的实现中,我没有看到对于这种差别的处理。验证的的方法是让父进程close
(fd【1】)后pause()。
总结:
管道在实际的应用中使用的比较广泛,对于pipe而言,尤其是在父子进程之间使用的较多,实
际上,pipe对于没有亲缘关系的进程来说,实现起来比较麻烦。没什么意义。希望各位朋友连
接管道的特性的话,看看上面的浅薄分析。或许对大家有用。
最近比较忙,有时间的话,我会适当的找些主题聊些自己的感想,欢迎各位朋友指正!
蓝色键盘
2003.6.20
另外,我觉得如下的这个帖子提出的问题相当的有讨论的价值,哪位有时间首先整理一下,或者说说自己的看法,大家一起讨论一下。
http://www.chinaunix.net/forum/viewtopic.php?t=94621
gadfly 回复于:2003-06-20 13:28:37
|
关于1.
[quote:fdfdb21490="蓝色键盘"]
综合以上三个系统来看,我们发现write数据到pipe,并不阻塞,不论这些数据有没有被其它的
进程读走。为了保证write彻底的将所有数据写完,可以让子进程sleep更长的时间。往一个管
道中写数据受到系统PIPE_BUF的限制。
[/quote:fdfdb21490]
不阻塞是什么意思?是我理解错了么?如果不阻塞,恐怕应该是parent先打印这行吧
parent[2003] process write: 100000
从结果来看,应该说是阻塞的,每次写入的数据,是受限于PIPE_BUF
|
gadfly 回复于:2003-06-20 13:44:15
|
至于2,补充一点:
先关闭读端再写的情况下,是这种结论。
但是以下这个帖子,
http://www.chinaunix.net/forum/viewtopic.php?t=93300
情况还不一样。引用我的理解:
[quote:aa417fa76c="gadfly"]
如果管道是在写的过程中关闭的,会返回成功的个数;如果在关闭后写,就会报pipe broken错,如果不捕捉或不阻塞或忽略这个SIGPIPE信号,缺省情况下,进程会直接退出,而不是write返回-1。
[/quote:aa417fa76c]
|
gadfly 回复于:2003-06-20 13:54:25
|
至于3,其实是1的延深。
还是基于我对测试1的理解,在最后一个PIPE_BUF大小的数据没读之前,父进程是阻塞,写,阻塞,写....的过程,也就是说实际上读剩最后的PIPE_BUF之后数据的时候,parent才写完,关闭写端。
但是这并不影响键盘兄关于这个问题的结论,
[quote:a432de988b="蓝色键盘"]
尽管管道的写端已经被关闭,然而在管道读端可用的情况下,内核保证管道中的数
据仍然能够被读取。
[/quote:a432de988b]
请各位指正。
|
shaoyijun 回复于:2003-06-20 14:06:09
|
[quote:cd07a287ba="gadfly"]ocess write: 100000
从结果来看,应该说是阻塞的,每次写入的数据,是受限于PIPE_BUF[/quote:cd07a287ba]
同意!!! :oops:
|
HopeCao 回复于:2003-06-20 14:06:14
|
应该是阻塞的吧!!!
[color=blue:d8ea25a484]"如果不阻塞,恐怕应该是parent先打印这行吧
parent[2003] process write: 100000
" [/color:d8ea25a484]
|
shaoyijun 回复于:2003-06-20 14:10:17
|
总结的好呀!!!
我昨天晚上也试了一下几种情况
发现自从到了chinaunix,我开始思考了!!! :oops:
|
gadfly 回复于:2003-06-20 14:11:47
|
[quote:a2f1dbe078="shaoyijun"]总结的好呀!!!
我昨天晚上也试了一下几种情况
发现自从到了chinaunix,我开始思考了!!! :oops:[/quote:a2f1dbe078]
那共享你的结论亚。
|
无双 回复于:2003-06-20 15:12:07
|
应该是阻塞
因为文件描述符初始时都是阻塞了
除非使用fcntl或是其它函数改变端口状态
|
蓝色键盘 回复于:2003-06-20 16:26:52
|
讨论的相当好,非常感谢各位,尤其是gadfly!当然,gadfly的回复是正确的。
看来,我们的理解是一致的,只是我对write的阻塞理解和大家存在差别,遗憾的是,我在总结中没有写上这种情况。
我总结的如下:
[code:1:c71875e438]
综合以上三个系统来看,我们发现write数据到pipe,并不阻塞,不论这些数据有没有被其它的进程读走。为了保证write彻底的将所有数据写完,可以让子进程sleep更长的时间。往一个管道中写数据受到系统PIPE_BUF的限制。
[/code:1:c71875e438]
如果定义宏WR<=PIPE_BUF的话,我们会看到,write会马上成功返回,并不阻塞。
实施上,正真的原因在于:
1、如果write的数据量>PIPE_BUF的话,write成功写的数据为PIPE_BUF。这时候只有写入的数据被读走。write才能继续写剩下的数据,我想大家认为的阻塞便是这种情况吧。但我总觉得理解为阻塞似乎不是很精确。不过也只能这么称呼。:P
2、如果write的数据量<=PIPE_BUF的话,write成功写入数据后返回。不会阻塞。而不管写入的数据有没有被读走。
欢迎指正!
|
蓝色键盘 回复于:2003-06-20 16:30:19
|
我的排版很乱,有时间重新处理一下.
:P
|
shaoyijun 回复于:2003-06-20 17:57:24
|
[quote:ddbe1e7379="gadfly"]
那共享你的结论亚。[/quote:ddbe1e7379]
我只是验证了一下4种情况,不是很深刻!
|
无双 回复于:2003-06-21 01:15:46
|
我觉得应该是总数据量
就是BUF中总数据量>-PIPE_LEN时会产生阻塞
而不是由当前WRITE的数据量决定
这个机制与TCP SOCKET是一样的
|
liujing6484 回复于:2004-06-01 12:28:31
|
我看了这些东西但是看不懂呀!真是还要好好学习!
|
原文转自:http://www.ltesting.net
- 评论列表(网友评论仅供网友表达个人看法,并不表明本站同意其观点或证实其描述)
-
|