这似乎是一件相当普遍的事情,而且我已经设法自学了我需要的一切,除了我现在有一个问题,这是我的故障排除.
@H_403_2@int nonBlockingPOpen(char *const argv[]){
int inpipe;
pid_t pid;
/* open both ends of pipe nonblockingly */
pid = fork();
switch(pid){
case 0: /*child*/
sleep(1); /*child should open after parent has open for reading*/
/*redirect stdout to opened pipe*/
int outpipe = open("./fifo",O_WRONLY);
/*SHOULD BLOCK UNTIL MAIN PROCESS OPENS FOR WRITING*/
dup2(outpipe,1);
fcntl(1,F_SETFL,fcntl(1,F_GETFL) | O_NONBLOCK);
printf("HELLO WORLD I AM A CHILD PROCESS\n");
/*This seems to be written to the pipe immediately,blocking or not.*/
execvp(*argv,argv);
/*All output from this program,which outputs "one" sleeps for 1 second
*outputs "two" sleeps for a second,etc,is captured only after the
*exec'd program exits!
*/
break;
default: /*parent*/
inpipe = open("./fifo",O_RDONLY | O_NONBLOCK);
sleep(2);
/*no need to do anything special here*/
break;
}
return inpipe;
}
为什么子进程每次生成一行时都不会将其stdout写入管道?在execvp或dup2的工作方式中是否有一些我缺少的东西?我知道我对这一切的处理方式有点奇怪,但我找不到另一种以编程方式捕获闭源二进制文件输出的方法.
解决方法
我猜你只有在退出后才能获得exec’d程序的输出,因为在每条消息之后它不会是
flush.如果是这样,你无法从外面做任何事情.
我不太确定这与你的问题中的阻塞和非阻塞I / O之间的选择有什么关系.非阻塞写入可能会完全或部分失败:不会阻塞程序,直到管道中有空间,调用立即返回并表示它无法写入应该拥有的所有内容.非阻塞I / O既不会使缓冲区变大,也不会强制刷新输出,并且某些程序可能会严重支持它.
你不能强制你正在执行的二进制程序刷新.如果您认为非阻塞I / O是该问题的解决方案,抱歉,但我担心这是非常正交的.
编辑:好吧,如果exec’d程序只使用libc提供的缓冲(没有实现它自己的)并且是动态链接的,你可以强制它通过将它与一个刷新每次写入的修改过的libc相连接来刷新.这将是一个绝望的措施.只有在其他一切都失败时才尝试.