深入解析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的用法及其在实际开发中的应用。希望本文对您有所帮助!

相关推荐
iuuia4 小时前
05--JavaScript基础语法(1)
开发语言·javascript·ecmascript
冲上云霄的Jayden4 小时前
ubuntu 22一步步 安装docker和配置使用国内源
linux·ubuntu·docker·国内源
一晌小贪欢4 小时前
Python爬虫第4课:XPath与lxml高级解析技术
开发语言·爬虫·python·网络爬虫·python爬虫·python3·python办公
蓝色汪洋4 小时前
string字符集
java·开发语言
虚行4 小时前
C#项目连接S7-PLCSIM Advanced读写操作
开发语言·python·c#
让我上个超影吧4 小时前
深入浅出 Java 中的 CompletableFuture:让异步编程变得简单
java·开发语言
CryptoPP4 小时前
获取越南股票市场列表(包含VN30成分股)实战指南
大数据·服务器·数据库·区块链
仰泳的熊猫4 小时前
LeetCode:889. 根据前序和后序遍历构造二叉树
数据结构·c++·算法
迎風吹頭髮4 小时前
Linux服务器编程实践20-TCP服务 vs UDP服务:核心差异对比
linux·服务器·tcp/ip