Linux 通过lseek()来实现文件大小的设置

时间:2022-09-05 18:16:13

一 函数介绍:

函数名: lseek()

功 能: 移动文件读/写指针

所需头文件:

#include <sys/types.h>

#include <unistd.h>

函数原型:

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

重新定位已打开的文件的偏移量,与whence的取值有关;

参数:

fd:文件描述符,对应已经打开的文件;

offset:指出偏移量;

whence:指出偏移的方式,取值如下:

SEEK_SET:偏移到offset位置处(相对文件头)

SEEK_CUR:偏移到当前位置+offset位置处;

SEEK_END:偏移到文件尾+offset位置处;

返回值:

调用成功则返回最终的偏移量(从文件头开始数);

调用失败则返回-1,并设置相应的errno;

二 巧妙利用

1. 返回当前的偏移量

off_t currpos;

currpos = lseek(fd, 0, SEEK_CUR);

2. 返回文件大小

off_t currpos;

currpos = lseek(fd, 0, SEEK_END);

3. 扩充文件大小

lseek()方法允许偏移

  这个技巧也可用于判断我们是否可以改变某个文件的偏移量。如果参数 fd(文件描述符)指定的是 pipe(管道)、FIFO 或者 socket,lseek 返回 -1 并且置 errno 为 ESPIPE。
  对于普通文件(regular file),cfo 是一个非负整数。但对于特殊设备,cfo 有可能是负数。因此,我们不能简单地测试 lseek 的返回值是否小于 0 来判断 lseek 成功与否,而应该测试 lseek 的返回值是否等于 -1 来判断 lseek 成功与否。
  lseek 仅将 cfo 保存于内核中,不会导致任何 I/O 操作。这个 cfo 将被用于之后的读写操作。

  如果 offset 比文件的当前长度更大,下一个写操作就会把文件“撑大(extend)”。这就是所谓的在文件里创造“空洞(hole)”。没有被实际写入文件的所有字节由重复的 0 表示。空洞是否占用硬盘空间是由文件系统(file system)决定的。

概念补充:

当前文件偏移量(current file offset),以下简称为 cfo。cfo 通常是一个非负整数,用于表明文件开始处到文件当前位置的字节数。读写操作通常开始于 cfo,并且使 cfo 增大,增量为读写的字节数。文件被打开时,cfo 会被初始化为 0,除非使用了 O_APPEND 。