如何在C中使用读写BUFSIZ

时间:2015-02-06 10:51:16

标签: c linux file system stdio

对于赋值,我应该创建两个方法:方法一将read()write()输入文件一个空的输出文件,一次一个字节(缓慢)。

另一种方法将使用char buf[BUFSIZ]; BUFSIZ来自<stdio.h>。我们应该read()write()使用BUFSIZ,这会让事情变得更快。

我们测试每个方法的输入文件只是一个linux字典(/dict/linux.words)。

我已经正确实现了方法一,我一次在一个字符上调用read()write(),将输入文件复制到输出文件。虽然速度非常慢,但它至少会复制一切。

我的代码如下:

// assume we have a valid, opened fd_in and fd_out file.
char buf;
while(read(fd_in, buf, 1) != 0)
    write(fd_out, buf, 1);

对于方法二,但是我使用BUFSIZ,我无法将每个条目传输到输出文件中。它在z条目中失败,并且不再写入。

所以,我的第一次尝试:

// assume we have a valid, opened fd_in and fd_out file
char buf[BUFSIZ];
while(read(fd_in, buf, BUFSIZ) != 0)
    write(fd_out, buf, BUFSIZ);

不起作用。

我知道read()将返回读取的字节数,如果它位于文件的末尾,则返回0。我遇到的问题是了解如何将read()BUFSIZ进行比较,然后循环并在其停止的位置启动read(),直到我到达文件的真实结尾。< / p>

2 个答案:

答案 0 :(得分:4)

由于您的文件很可能不是BUFSIZ的精确倍数,因此您需要检查读取的实际字节数,以便最后一个块被正确写入,例如

char buf[BUFSIZ];
ssize_t n;
while((n = read(fd_in, buf, BUFSIZ)) > 0)
    write(fd_out, buf, n);

答案 1 :(得分:0)

this code:

// assume we have a valid, opened fd_in and fd_out file
char buf[BUFSIZ];
while(read(fd_in, buf, BUFSIZ) != 0)
    write(fd_out, buf, BUFSIZ);

leaves much to be desired, 
does not handle a short remaining char count at the end of the file, 
does not handle errors, etc.

a much better code block would be:

// assume we have a valid, opened fd_in and fd_out file
char buf[BUFSIZ];
int readCount;  // number of bytes read
int writeCount; // number of bytes written

while(1)
{
    if( 0 > (readCount = read(fd_in, buf, BUFSIZ) ) )
    { // then, read failed
         perror( "read failed" );
         exit( EXIT_FAILURE );
    }

    // implied else, read successful

    if( 0 == readCount )
    {  // then assume end of file
        break; // exit while loop
    }

    // implied else, readCount > 0

    if( readCount != (writeCount = write( fd_out, buf, readCount ) ) )
    { // then, error occurred
        perror( "write failed" );
        exit( EXIT_FAILURE );
    }

    // implied else, write successful
} // end while

注意:我没有包括输入/​​输出文件语句的关闭          在每次调用exit()之前,确实需要添加