文件I/O读写操作

在上一篇文章中,我简单讲述了如何进行打开文件操作。而在这一篇文章中,我将会对打开的文件进行操作。
在进行文件读写操作之前,需要了解一个缓冲区(buff)的概念,缓冲区是在内存中预留指定大小的存储空间用来对I/O的数据做临时存储,这部分预留的内存空间叫缓冲区。说白了就是内核把数据从文件中读出来,没地方放,只能在内存中开辟一块临时的空间去存放这个数据,直到这个数据再次被操作,这时候缓冲区中的数据就被拿走了。
下面是read函数和write函数,这两个函数参数差不多,我把他们并在一起讲下面先看函数定义:

ssize_t read(int fd, void *buf, size_t count);
ssize_t write(int fd, const void *buf, size_t count);

是不是非常类似,在这里ssize_t是经过类型重定义的int型。
对于第一个参数fd,就是之前使用open函数打开的文件描述符,第二个参数就是最上面所说的缓冲区,我们需要自己定义一个缓冲区buff,一般都是
char buff[buffsize]=”\0”;在这里我把缓冲区buff给初始化了,这是为了防止缓冲区原本有乱码。第三个参数是要读字节的大小,一般来说我们直接填sizeof(buff)就好了。
他们的返回值都是已近操作的字节数(读入的,读出的),而错误都是返回-1。在这里我写一段代码用于计算文件大小。已知文件描述符fd,打开的文件中有字。

#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <sys/stat.h>

#define N 64

int main(int argc,char *argv[])
{
    int fd,sum=0;
    int ret;
    char buff[N] = "\0";
    while((ret = read(fd,buff,sizeof(buff)))>0)
    //循环从文件中读取数据,直至文件尾部
    {
        sum += ret;//每次读取计数
    }
    printf("the length of the txt is:%d\n",sum);
    close(fd);
    return 0;
}

这样子就完成了对一个文件中字节的计数。
下面我要讲下lseek函数,即文件偏移量(current file offset),读/写操作是从当前文件偏移处开始的,在成功进行读/写操作后,文件偏移量增加,增加量是进行读/写的字节数。当文件刚刚打开时,文件偏移量被初始化为0。还记得之前的O_APPEND文件打开参数,则每次写操作前,将文件位移量设置在文件的当前结尾处。

off_t lseek(int fd, off_t offset, int whence);

off_t也是类型重定义后int类型,fd是文件描述符,offset是对于基准点的偏移量,可正可负(向前移,向后移),whence是基准点,一般是宏定义,但是也可以写成数字的形式。有以下三个基准点:
SEEK_SET:代表文件起始位置,数字表示为0
SEEK_CUR:代表文件当前的读写位置,数字表示为1
SEEK_END:代表文件结束位置,数字表示为2
最后,lseek函数返回值成功就是当前文件偏移量,失败返回-1。
这里需要注意的是:
lseek函数不直接进行任何I/O操作,而是把数据记录在内核中。而且该函数只对常规文件有效,对于管道,FIFO等无法进行操作。标准I/O中有个fseek与之类似。

最后再补充下文件偏移量实际中的应用:
文件位移量可以大于文件的当前长度,在这种情况下,对该文件的写操作会延长文件,并形成空洞。
在UNIX/Linux文件操作中,文件位移量可以大于文件的当前长度,在这种情况下,对该文件的下一次写将延长该文件,并在文件中构成一个空洞,这一点是允许的。位于文件中但没有写过的字节都被设为’\0’,用read读取空洞部分读出的数据是‘\0’。
空洞文件作用很大,例如迅雷下载文件,在未下载完成时就已经占据了全部文件大小的空间,这时候就是空洞文件。下载时如果没有空洞文件,多线程下载时文件就都只能从一个地方写入,这就不是多线程了。如果有了空洞文件,可以从不同的地址写入,就完成了多线程的优势任务。
Tips:可以使用ls -lh 和 du -sh 对比看空洞文件和正常文件的区别

  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值