10

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

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);
}

这是从管道中逐行读取最佳值的正确方法吗?

4

2 回答 2

19

这通常被称为从 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不会写入超过传递给它的大小。如上所述,另一半正在对可能由意外长的输入行导致的部分行做一些明智的事情。

于 2013-04-23T07:23:58.187 回答
1

这是另一种选择(我不完全确定这是一种“正确”的方式)-使用read函数读取的字节数。在此示例中,stdin尽管进行了重定向,但我正在阅读,因此 0 中的 fd 是文件/管道/您需要的任何内容。

  while ((nbytes=read(STDIN_FILENO, buffer, MAX_PIPE_SIZE)) > 0) {
    write(STDOUT_FILENO, buffer, nbytes);
  }
于 2015-12-16T23:01:46.473 回答