关闭

linux程序设计——管道输出数据到popen(第十三章)

826人阅读 评论(0) 收藏 举报
分类:

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命令输出的长度,这种情况就会发生.

0
0

查看评论
* 以上用户言论只代表其个人观点,不代表CSDN网站的观点或立场
    个人资料
    • 访问:109212次
    • 积分:3372
    • 等级:
    • 排名:第10304名
    • 原创:232篇
    • 转载:5篇
    • 译文:0篇
    • 评论:10条
    最新评论