笔记3-4: lseek 函数
lseek 函数:
原型:
#include <unistd.h>
off_t lseek(int filedes, off_t offset, int whence);
filedes 为文件描述符。
offset带符号的为偏移量,off_t有可能是32位,也有可能是64位。
当whence是SEEK_SET,则将偏移量设置为距离文件开始处offset个字节。
当whence是SEEK_CUR,则将该文件的当前偏移量的值加offset, offset可正可负。
当whence是SEEK_END,则将该文件的偏移量设置为文件长度的值加offset, offset可正可负。
若lseek执行成功,则返回新的文件偏移量。
可以使用下面方法获取文件当前偏移量:
currpos = lseek(fd, 0, SEEK_CUR);
可以使用这种方法来确定当前的描述符是否可以设置偏移量。如果文件描述符引用的是一个管道、FIFO、网络套接字,则lseek会返回-1,并设置errno为ESPIPE。
实例1:测试标准输入是否可以设置偏移量
#include “apue.h”
int main(void)
{
if (lseek(STDIN_FILENO, 0, SEEK_CUR) == -1)
printf(“cannot seek \n”);
else
printf(“seek ok. \n”);
}
lseek仅将当前的文件偏移量记录在内核中,它并不会引起任何的I/O操作,但是lseek会影响到下一个读与写的操作。
文件偏移量可以大于文件的长度,这种情况下,如果对文件的下一次写操作,将会加长该文件,并在文件中构成一个空洞,这是允许的,空洞中的未写过的字节都视为0。空洞并不一定占用磁盘的空间,这一点与文件系统的具体实现有关。
实例2:创建有空洞的文件
#include “apue.h”
#include <fcntl.h>
char buf1[] = “abcdefghij”;
char buf2[] = “ABCDEFGHIJ”;
int main(void)
{
int fd;
if ((fd = creat(“file.hole”, FILE_MODE)) < 0)
err_sys(“creat error”);
if (write(fd, buf1, 10) != 10)
err_sys(“buf1 write error”);
// offset now == 10
if (lseek(fd, 16384, SEEK_SET) == -1)
err_sys(“lseek error”);
// offset now == 16384
if (write(fd, buf2, 10) != 10)
err_sys(“buf2 write error”);
// offset now == 16394
return 0;
}