13.3 将输出送往popen
看过捕获外部程序输出的例子后,再来看一个将输出发送到外部程序的示例程序popen2.c,它将数据通过管道送到另一个程序.在这里使用的是od(八进制)命令.
编写程序popen2.c,它非常类似popen1.c,唯一的不同是这个程序将数据写入管道,而不是从管道中读取.
/************************************************************************* > File Name: popen2.c > Description: popen2.c程序将数据写入管道 > Author: Liubingbing > Created Time: 2015年07月09日 星期四 19时43分51秒 > Other: popen2.c程序的作用等同于在命令行上输入echo "Once upon a time, there was..." | od -c ************************************************************************/ #include <stdio.h> #include <stdlib.h> #include <stdio.h> #include <string.h> int main(){ FILE *write_fp; char buffer[BUFSIZ + 1]; /* 将字符串写入buffer */ sprintf(buffer, "Once upon a time, there was...\n"); /* 函数popen允许一个程序给另一个程序传输数据 * 第二个参数"w"表明调用程序可以用fwrite调用向被调用程序发送数据 * 即将文件流指针write_fp指向的数据作为命令"od -c"的标准输入 */ write_fp = popen("od -c", "w"); if (write_fp != NULL) { /* fwrite向一个文件写入数据块 * 即将buffer中的数据写入到write_fp指向的文件,写入strlen(buffer)个元素,每个元素的大小为sizeof(char)个字节 */ fwrite(buffer, sizeof(char), strlen(buffer), write_fp); pclose(write_fp); exit(EXIT_SUCCESS); } exit(EXIT_FAILURE); }
程序使用带有参数"w"的popen启动od -c命令,这样就可以向该命令发送数据了.然后它给od -c命令发送一个字符串,该命令接收并处理它,最后把处理结果打印到自己的标准输出上.
在命令行上,可以用下面的命令得到同样的输出结果:
$ echo "Once upon a time, there was..." | od -c
程序popen2.c的结果如下所示:
13.3.1 传递更多的数据
目前所使用的机制都只是将所有数据通过一次fread或fwrite调用来发送或接收.有时,可能希望以块方式发送数据,或者根本就不知道输出数据的长度.为了避免定义一个非常大的缓冲区,可以用多个fread或fwrite调用来将数据分为几部分处理.
编写程序popen3.c,通过管道读取所有数据
/************************************************************************* > File Name: popen3.c > Description: popen3.c程序通过管道读取大量数据 > Author: Liubingbing > Created Time: 2015年07月09日 星期四 20时34分37秒 > Other: popen3.c ************************************************************************/ #include <stdio.h> #include <stdlib.h> #include <string.h> #include <unistd.h> int main(){ FILE *read_fp; char buffer[BUFSIZ + 1]; int chars_read; memset(buffer, '\0', sizeof(buffer)); /* popen函数访问"ps ax"命令给出的信息 * 第一个参数"ps ax"是要运行的程序名和相应的参数 * 第二个参数open_mode="r"说明被调用程序的输出可以被调用程序使用 * 返回值为FILE*文件流指针read_fp,利用read_fp可以读取"ps ax"的输出信息 */ read_fp = popen("ps ax", "r"); if (read_fp != NULL) { /* fread函数从文件流指针read_fp指向的文件流中读取数据 * 最多读取BUFSIZ个元素,每个元素sizeof(char)个字节 * buffer用于接收数据的内存地址 * 如果成功则返回实际读取的元素的个数 */ chars_read = fread(buffer, sizeof(char), BUFSIZ, read_fp); if (chars_read > 0) { buffer[chars_read - 1] = '\0'; printf("Reading %d:-\n %s\n", BUFSIZ, buffer); chars_read = fread(buffer, sizeof(char), BUFSIZ, read_fp); } /* pclose关闭与popen关联的文件流(read_fp指向的文件流) */ pclose(read_fp); exit(EXIT_SUCCESS); } exit(EXIT_FAILURE); }
在这个程序中,从被调用的进程ps ax中读取数据.该进程输出的数据有多少事先无法知道,所以必须对管道进行多次读取.
这个程序调用popen函数时使用了"r"参数,这与popen1.c程序做法一样.这次,它连续从文件流中读取数据,直到没有数据可读为止.注意,虽然ps命令的执行要花费一些时间,但linux会安排好进程间的调度,让两个程序在可以运行时继续运行.如果读进程popen3没有数据可读,它将被挂起直到有数据到达.如果写进程ps产生的数据超过了可用缓冲区的长度,它也会被挂起直到读进程读取了一些数据.
在本例中,可能不会看到Reading:-信息的第二次出现.如果BUFSIZ的值超过了ps命令输出的长度,这种情况就会发生.
版权声明:本文为博主原创文章,未经博主允许不得转载。