Bootstrap

Linux系统调用之lseek函数

前言

如果,想要深入的学习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函数的功能还挺好玩的。

;