前言
如果,想要深入的学习Linux系统调用函数lseek了话,还是需要去阅读Linux系统中的帮助文档的。
具体输入命令:
man 2 lseek
即可查阅到完整的资料信息。
lseek函数
lseek函数是Linux系统API中的一员,它的官方定义是:重新定位读或写的文件偏移量。
这里科普一下什么叫做当前文件偏移量:
- 每当打开一个文件,都会有一个叫做“当前文件偏移量”的东西,如果难理解也可以将他理解为指针。 除非打开文件时指定O_APPEND选项,否则文件偏移量默认设置为0。当我们发生了一次读或者写操作时,都会使这个当前文件偏移量发生变化,读/写多少字节,当前偏移量就会向后移动多少。
知道了这个概念了后,我们就了解了lseek函数它有些什么作用。下面我们来细细介绍一下这个函数。
它的函数原型是长这样的:
off_t lseek(int fd, off_t offset, int whence);
先来说一下这个off_t类型吧,它用于指示文件的偏移量。你可以就简单的理解为这是一个64位的整形数,相当于long long int,其定义在unistd.h头文件中可以查看。
在使用这个函数之前,我们需要往C/C++文件中导入这些头文件:
#include <sys/types.h>
#include <unistd.h>
通过lseek函数的函数原型我们可以知道,我们需要给它传入3个参数,那我们依次介绍这三个参数是什么,有什么含义在里面。
- PS:题外话,如果有听不懂的地方,请自行查阅Linux帮助文档,开头有些查阅方法,一手知识永远是最好的知识。
参数:fd //文件描述符,可以通过open函数得到,通过这个fd可以操作某个文件
参数: offset //文件偏移量,是一个整形数
参数:whence //偏移类型,下列三个值中选一个。whence :
SEEK_SET:该文件的偏移量设为离文件开头offset个字节.
SEEK_CUR:该文件的偏移量设为其当前值加offset(PS :offest可正负).
SEEK_END:该文件的偏移量设为文件长度加offset
再来聊一下返回值:
如果成功:返回文件指针的位置
如果失败:返回-1,并将错误原因赋值给errno,我们可以用标准C库中的perror函数打印出错误原因。需要引入头文件 #include <stdio.h>
lseek函数的作用
lseek函数的作用有以下四点:
-
1.移动文件指针到文件头:
lseek(fd, 0, SEEK_SET);
-
2.获取当前文件指针的位置
lseek(fd, 0, SEEK_CUR);
-
3.获取文件长度
lseek(fd, 0, SEEK_END);
-
4.拓展文件的长度,当前文件10b, 110b, 增加了100个字节
lseek(fd, 100, SEEK_END)
注意:拓展完需要再写一次数据,否则拓展无效
光这样介绍不可能学的会,我们来通过一个实战例子来彻底了解一下这个lseek函数
实战演练:lseek函数
作用1:移动文件指针到文件头:
//导入所有需要的头文件
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
//我们的目的是:移动文件指针到文件头:
int main()
{
//获取文件的文件描述符
int fd = open("text.txt", O_RDWR);
if (fd == -1)
{
perror("open");
return -1;
}
//输出当前文件的偏移量
long long int loc = lseek(fd, 0, SEEK_CUR);
printf("%lld\n", loc);
//使用read函数读3个字节的数据
char buf[3] = {0};
int rnum = read(fd, buf, sizeof(buf));
printf("%d\n", rnum);
//再次查看文件的偏移量
long long int loc1 = lseek(fd, 0, SEEK_CUR);
printf("%lld\n", loc1);
//移动文件指针到文件头
long long int loc2 = lseek(fd, 0, SEEK_SET);
printf("%lld\n", loc2);
return 0;
}
作用2:获取当前文件指针的位置
//导入所有需要的头文件
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
int main()
{
//导入所有需要的头文件
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
int main()
{
//获取文件的文件描述符
int fd = open("hello.txt", O_RDWR);
//输出当前文件的偏移量
long long int loc = lseek(fd, 0, SEEK_CUR);
printf("%lld\n", loc);
//使用read函数读2个字节的数据
char buf[2] = {0};
int rnum = read(fd, buf, sizeof(buf));
printf("%d\n", rnum);
//再次查看文件的偏移量
long long int loc1 = lseek(fd, 0, SEEK_CUR);
printf("%lld\n", loc1);
return 0;
}
}
作用3:获取文件长度
//导入所有需要的头文件
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
int main()
{
//获取文件的文件描述符
int fd = open("hello.txt", O_RDWR);
//获取文件长度
long long int loc1 = lseek(fd, 0, SEEK_END);
printf("%lld\n", loc1);
return 0;
}
作用4:拓展文件的长度(注:拓展完需要再写一次数据,否则拓展无效)
//导入所有需要的头文件
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
int main()
{
//获取文件的文件描述符
int fd = open("hello.txt", O_RDWR);
//获取文件长度
long long int loc1 = lseek(fd, 0, SEEK_END);
printf("%lld\n", loc1);
//拓展文件的长度
long long int loc2 = lseek(fd, 100, SEEK_END);
write(fd," ",1);//写入一个空数据
printf("%lld\n", loc2);
return 0;
}
总结
当明白了当前文件偏移量这个概念了以后,lseek函数也变的并不是那么难理解了。仔细试了一下lseek函数的功能还挺好玩的。