在C中逐行读取管道

时间:2013-04-23 07:12:54

标签: c pipe stdout stdin

如何分隔来自管道的线条。在管道中有这样的文字:

HALLO:500\n
TEST:300\N
ADAD
ADAWFFA
AFAGAGAEG

我想将线与管道分开,因为我想将值保存在变量中。

这是我的c代码:

#include <stdio.h>
#include <stdlib.h>

#define BUFFERSIZE    1

int main(int argc, char **argv){
    unsigned char     buffer[BUFFERSIZE];
    FILE                         *instream;
    int                            bytes_read=0;
    int                            buffer_size=0;


    buffer_size=sizeof(unsigned char)*BUFFERSIZE;
    /* open stdin for reading */
    instream=fopen("/dev/stdin","r");

    /* did it open? */
    if(instream!=NULL){
        /* read from stdin until it's end */
        while((bytes_read=fread(&buffer, buffer_size, 1, instream))==buffer_size){
            fprintf(stdout, "%c", buffer[0]);
        }
    }
    /* if any error occured, exit with an error message */
    else{
        fprintf(stderr, "ERROR opening stdin. aborting.\n");
        exit(1);
    }

    return(0);
}

这是从管道读取最佳线路的正确方法吗?

2 个答案:

答案 0 :(得分:14)

这通常被称为从stdin读取。程序不应该关心输入是管道,重定向文件还是键盘。

fread只会读取,直到缓冲区已满。使用fgets读取一行。

此外,缓冲区大小应足够大以容纳线路。对于一次性小程序,你可以选择一个号码。或者有一个标准名称BUFSIZ,它为您提供了一个非常大的缓冲区。多大?足够大。真?可能。

fgets将复制字符串中的换行符,除非字符串先填满。因此,您可以测试最后一个字符,以判断该行是否被截断。有了合理的投入,那就不会发生。但是更强大的方法是分配更大的缓冲区,复制部分行,再次调用fgets tp继续尝试获得完整的行。

#include <stdio.h>

int main() {
    char buf[BUFSIZ];
    fgets(buf, sizeof buf, stdin);
    if (buf[strlen(buf)-1] == '\n') {
        // read full line
    } else {
        // line was truncated
    }
    return 0;
}

这可以帮助您免受可怕的缓冲区溢出问题的影响。 fgets写入的大小不会超过传递给它的大小。如上所述,另一半是对可能由于意外长输入行可能导致的部分线做了明智的事情。

答案 1 :(得分:1)

这是另一种选择(我不完全确定它是'正确'的方式) - 使用read函数读取的字节数。 在这个例子中,我是从stdin读取的,虽然进行了重定向,因此0中的fd是文件/管道/无论你需要它。

  while ((nbytes=read(STDIN_FILENO, buffer, MAX_PIPE_SIZE)) > 0) {
    write(STDOUT_FILENO, buffer, nbytes);
  }