深入解析Linux下的`lseek`函数:文件定位与操作的艺术

深入解析Linux下的`lseek`函数:文件定位与操作的艺术

在Linux系统编程中,lseek函数是一个不可或缺的工具,它允许开发者精确控制文件指针的位置,从而实现高效的文件读取、写入和扩展操作。本文将深入探讨lseek函数的各个方面,包括其功能、使用场景、实际应用案例以及与其他类似函数的比较,帮助开发者全面掌握这一重要工具。


1. lseek函数的功能与参数

lseek函数用于改变与文件描述符关联的文件指针的位置。其函数原型如下:

c 复制代码
#include <sys/types.h>
#include <unistd.h>

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

参数说明:

  • fd :文件描述符,必须有效且已打开。
  • offset :偏移量,表示从whence指定的起始点开始移动的距离。
  • whence :指定起始点,取值为:
    • SEEK_SET:文件开头。
    • SEEK_CUR:当前文件指针位置。
    • SEEK_END:文件末尾。

返回值:

  • 成功时,返回新的文件指针位置(以字节为单位)。
  • 失败时,返回-1,并设置errno

错误处理:

  • EBADF:文件描述符无效。
  • EINVALwhence参数无效。
  • ESPIPE:尝试对管道、套接字或类似设备进行写入操作。

2. 写完文件再读

在Linux中,写完文件后,文件指针位于文件末尾。如果需要读取文件内容,必须先调整文件指针的位置。

示例代码:

c 复制代码
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>

int main() {
    int fd = open("test.txt", O_WRONLY | O_CREAT | O_TRUNC, 0644);
    if (fd == -1) {
        perror("open");
        exit(EXIT_FAILURE);
    }

    // 写入数据
    char* data = "Hello, World!";
    write(fd, data, strlen(data));

    // 写完后,文件指针在末尾,无法直接读取
    // 需要调整文件指针到开头
    if (lseek(fd, 0, SEEK_SET) == -1) {
        perror("lseek");
        close(fd);
        exit(EXIT_FAILURE);
    }

    // 以读模式打开文件(注意:必须重新打开文件,因为写模式下无法读取)
    close(fd);
    fd = open("test.txt", O_RDONLY);
    if (fd == -1) {
        perror("open");
        exit(EXIT_FAILURE);
    }

    // 读取数据
    char buffer[1024];
    int bytes_read = read(fd, buffer, sizeof(buffer)-1);
    if (bytes_read == -1) {
        perror("read");
        close(fd);
        exit(EXIT_FAILURE);
    }
    buffer[bytes_read] = '\0';
    printf("Read: %s\n", buffer);

    close(fd);
    return 0;
}

注意事项:

  • 写入文件后,文件描述符处于写模式,无法直接读取。需要重新以读模式打开文件。
  • 使用lseek调整文件指针位置时,必须确保文件描述符有效。

3. 使用lseek读取文件大小

lseek可以用来获取文件的大小。通过将文件指针移动到文件末尾,可以得到文件的总字节数。

示例代码:

c 复制代码
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>

int main() {
    int fd = open("test.txt", O_RDONLY);
    if (fd == -1) {
        perror("open");
        exit(EXIT_FAILURE);
    }

    // 移动文件指针到文件末尾
    off_t file_size = lseek(fd, 0, SEEK_END);
    if (file_size == -1) {
        perror("lseek");
        close(fd);
        exit(EXIT_FAILURE);
    }

    printf("File size: %ld bytes\n", file_size);

    close(fd);
    return 0;
}

注意事项:

  • 文件必须以读模式打开,否则可能导致错误。
  • lseek返回的是文件指针的新位置,即文件的总大小。

4. 使用lseek扩展文件大小

可以通过lseek将文件指针移动到所需位置,然后写入数据来扩展文件大小。

示例代码:

c 复制代码
#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>
#include <stdlib.h>

int main() {
    int fd = open("test.txt", O_WRONLY | O_CREAT | O_TRUNC, 0644);
    if (fd == -1) {
        perror("open");
        exit(EXIT_FAILURE);
    }

    // 写入初始数据
    char* data = "Hello, World!";
    write(fd, data, strlen(data));

    // 移动文件指针到新的位置(扩展文件)
    off_t new_size = 1024; // 扩展到1024字节
    if (lseek(fd, new_size - 1, SEEK_SET) == -1) {
        perror("lseek");
        close(fd);
        exit(EXIT_FAILURE);
    }

    // 写入一个字节来扩展文件
    char zero = '\0';
    write(fd, &zero, 1);

    close(fd);
    return 0;
}

注意事项:

  • 移动文件指针到new_size - 1,然后写入一个字节,以确保文件大小扩展到new_size
  • 文件必须以写模式打开,否则可能导致错误。

5. 与标准库fseek函数的比较

fseek是C标准库函数,用于调整文件指针的位置。以下是lseekfseek的比较:

属性 lseek fseek
参数 int fd FILE *stream
返回值 off_t int
文件描述符 使用文件描述符 使用FILE指针
处理方式 系统调用 库函数
错误处理 设置errno 返回非零值表示错误

示例代码:

c 复制代码
#include <stdio.h>
#include <fcntl.h>
#include <unistd.h>

int main() {
    int fd = open("test.txt", O_RDWR | O_CREAT | O_TRUNC, 0644);
    if (fd == -1) {
        perror("open");
        exit(EXIT_FAILURE);
    }

    // 使用`lseek`
    if (lseek(fd, 10, SEEK_SET) == -1) {
        perror("lseek");
        close(fd);
        exit(EXIT_FAILURE);
    }

    // 使用`fseek`
    FILE *fp = fdopen(fd, "r+");
    if (fseek(fp, 10, SEEK_SET) != 0) {
        perror("fseek");
        fclose(fp);
        exit(EXIT_FAILURE);
    }

    fclose(fp);
    close(fd);
    return 0;
}

注意事项:

  • lseek是系统调用,性能更高,但需要处理文件描述符。
  • fseek是库函数,使用更方便,但性能稍低。

总结

lseek函数在Linux系统编程中是一个非常有用的工具,能够帮助开发者精确控制文件指针的位置。通过本文的介绍,读者可以更好地理解lseek的用法及其在实际开发中的应用。希望本文对您有所帮助!

相关推荐
SPC的存折6 分钟前
(自用)LNMP-Redis-Discuz5.0部署指南-openEuler24.03-测试环境
linux·运维·服务器·数据库·redis·缓存
舒一笑7 分钟前
Docker Compose 挂载 Nginx 配置的正确姿势(90%的人都踩过这个坑)
运维·docker·容器
海寻山9 分钟前
Java内部类:4种类型+实战场景+面试避坑
java·开发语言·面试
W.W.H.13 分钟前
嵌入式常见面试题——操作系统与RTOS篇
linux·经验分享·操作系统·rtos
梦游钓鱼15 分钟前
stl常用容器说明
开发语言·c++
云飞云共享云桌面18 分钟前
共享云主机告别传统电脑——制造工厂研发部门2台三维设计云主共享给20个设计师并发用
大数据·运维·服务器·自动化·电脑·制造
航Hang*19 分钟前
Windows Server 配置与管理——第10章:配置FTP服务器
运维·服务器·网络·windows·学习·vmware
此刻觐神22 分钟前
IMX6ULL开发板学习-05(Linux之Vi/Vim编辑器的使用)
linux·学习·编辑器
WBluuue23 分钟前
Codeforces 1088 Div1+2(ABC1C2DEF)
c++·算法
摩西蒙24 分钟前
软考计算机组成原理学习笔记-1
笔记·学习·软件工程