Linux系统调用四、lseek()函数详解

简介: Linux系统调用四、lseek()函数详解

❀1. 案例:写文件并把写入内容打屏

我们可以通过上一节所讲的read()和write()函数来实现向一个文件中写入内容并把写入内容打印到屏幕的功能。

/************************************************************
  >File Name  : readandprint.c
  >Author     : QQ
  >Company    : QQ
  >Create Time: 2022年05月13日 星期五 12时11分44秒
************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>
#define BUF_MAX 512 /*buf缓冲区最大值*/
/*向中文件写入数据并把写入内容打印到标准输出*/
int main(int argc, char* argv[])
{
  if(argc < 2)
  {
    printf("not fount file name");
    return -1;  
  }
  int fd = open(argv[1], O_RDWR | O_CREAT);
  write(fd, "hello linux...", 15);
  char buf[20];
  memset(buf, 0, sizeof(buf));
  int read_size = read(fd, buf, sizeof(buf));
  if(read_size > 0)
  {
    write(STDOUT_FILENO, buf, read_size); /*STDIN_FILENO STDERR_FILENO*/
  }
  close(fd);
  return 0;
}

我们知道,在C语言中,字符串都是以 ‘\0’ 结尾的,比如 “hello linux…” 加上结束符共15字节。

write(fd, "hello linux...", 15);

我们来测试下程序,首先明确一点,字符串会写入相应文件,但是不会打印在屏幕中,这个后面分析。这里先看一下结束符 ‘\0’ 是如何显示的。

可以看到,确实不会打屏,且文件内容已写入。我们通过vim编辑器打开1.txt文件。

可以看到一个 ‘^@’ 字符,这个就是我们多写入的 ‘\0’ 字符,如果我们把写入字节数15改为14,就没有这个字符了。

❀2. lseek移动文件读写位置

  • 包含头文件
#include <sys/types.h>
#include <unistd.h>
  • 函数原型
off_t lseek(int fd, off_t offset, int whence);
  • 函数功能
    reposition read/write file offset.
  • 函数参数
  • fd:文件描述符
  • offset:偏移量
  • whence:位置
  • SEEK_SET:The offset is set to offset bytes. offset为0时表示文件开始位置。
  • SEEK_CUR:The offset is set to its current location plus offset bytes. offset为0时表示当前位置。
  • SEEK_END:The offset is set to the size of the file plus offset bytes. offset为0时表示结尾位置
  • 函数返回值
  • 成功返回当前位置到开始的长度
  • 失败返回-1并设置errno

下面我们通过上面的案例来分析lseek函数的用法,上面案例测试中说到,字符串已经写入了相应文件,但是并没有打印在屏幕中。这是因为,我们用write()函数写入文件之后,这时候读写位置就指在写完后的那个位置,也就是字符串的后面,这样我们在使用read()函数去读的时候就相当于从写入字符串的后面去读的,所以啥也没读到。这时候,就可以使用lseek()函数来移动读写位置,我们只需在上面代码中加一句话即可。

#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>
#define BUF_MAX 512 /*buf缓冲区最大值*/
/*向中文件写入数据并把写入内容打印到标准输出*/
int main(int argc, char* argv[])
{
  if(argc < 2)
  {
    printf("not fount file name");
    return -1;  
  }
  int fd = open(argv[1], O_RDWR | O_CREAT);
  write(fd, "hello linux...", 15);
    /*读写位置在末尾*/
    /*把读写位置移动到文件首部*/
    lseek(fd, 0, SEEK_SET);
  char buf[20];
  memset(buf, 0, sizeof(buf));
  int read_size = read(fd, buf, sizeof(buf));
  if(read_size > 0)
  {
    write(STDOUT_FILENO, buf, read_size); /*STDIN_FILENO STDERR_FILENO*/
  }
  close(fd);
  return 0;
}

再测试一下,就发现可以正常打屏了。

❀3. lseek计算文件大小

利用lseek()函数执行成功时的返回值可以来计算一个文件所占字节的大小。

/************************************************************
  >File Name  : getsize.c
  >Author     : QQ
  >Company    : QQ
  >Create Time: 2022年05月13日 星期五 18时47分04秒
************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
int main(int argc, char* argv[])
{
  if(argc < 2)
  {
    printf("not found filename\n");
    return -1;  
  }
  int fd = open(argv[1], O_RDONLY);
  int size = lseek(fd, 0, SEEK_END);
  printf("file size: %d\n", size);
  close(fd);
  return 0;
}

运行程序测试结果如下。

❀4. lseek拓展文件大小

我们知道lseek()函数有三个参数,在前面的案例中,都把第二个参数偏移量offset设置为0来处理的,这样第三个参数就不用加偏移量了,相当于whence位置都是相对于文件首部来计算的。如果我们使用第二个参数offset,并把位置whence设置为文件尾,就相当于在文件尾再偏移offset个字节,这就达到了扩展文件大小的目的。

/************************************************************
  >File Name  : expandfile.c
  >Author     : QQ
  >Company    : QQ
  >Create Time: 2022年05月13日 星期五 19时02分06秒
************************************************************/
#include <stdio.h>
#include <stdlib.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <errno.h>
int main(int argc, char* argv[])
{
  if(argc < 2)
  {
    perror("not found filename: ");
    return -1;  
  }
  int fd = open(argv[1], O_WRONLY);
  lseek(fd, 10, SEEK_END);
  close(fd);
  return 0;
}

编译并运行,然后查看文件大小是否增加。

通过对比我们发现,文件大小并未增加。这是因为通过lseek()扩展了文件的大小之后,如果我们没有对该文件进行写操作,那么这个扩展的内容默认是不会保存的,所以文件大小不会改变。所以,在扩展后,至少要对文件写一次才能保存,我们对上面程序增加一个写操作,然后进行测试。

int main(int argc, char* argv[])
{
  if(argc < 2)
  {
    perror("not found filename: ");
    return -1;  
  }
  int fd = open(argv[1], O_WRONLY);
  lseek(fd, 10, SEEK_END);
    write(fd, "a", 1);
  close(fd);
  return 0;
}

我们运行后发现,文件大小从0变成了11,扩展了11个字节,而我们程序中仅指定扩展了10个字节,这是因为我们扩展完后又写入了一个字节a,通过前面的分析我们知道,在lseek()函数执行完毕后,读写位置应该是在文件尾部,这时再写入一个字符就相当于在文件尾部,也就是第11个字节出写入了一个a,保存后最终大小为11字节。我们可以使用vim打开文件查看一下。

可以看到10个 ‘^@’ 字符,第11个字符为写入的 ‘a’ 。

注:这里用到了一个函数叫做perror(),这个函数是用来打印错误信息的,我们在上面这些函数的返回值都可以看到一条,如果出错会设置errno,而设置errno就是和perror()函数相关联的,通过perror()这个函数就可以把出错信息打印出来。


相关文章
|
1月前
|
网络协议 Linux 调度
深入探索Linux操作系统的心脏:内核与系统调用####
本文旨在揭开Linux操作系统中最为核心的部分——内核与系统调用的神秘面纱,通过生动形象的语言和比喻,让读者仿佛踏上了一段奇妙的旅程,从宏观到微观,逐步深入了解这两个关键组件如何协同工作,支撑起整个操作系统的运行。不同于传统的技术解析,本文将以故事化的方式,带领读者领略Linux内核的精妙设计与系统调用的魅力所在,即便是对技术细节不甚了解的读者也能轻松享受这次知识之旅。 ####
|
29天前
|
缓存 算法 安全
深入理解Linux操作系统的心脏:内核与系统调用####
【10月更文挑战第20天】 本文将带你探索Linux操作系统的核心——其强大的内核和高效的系统调用机制。通过深入浅出的解释,我们将揭示这些技术是如何协同工作以支撑起整个系统的运行,同时也会触及一些常见的误解和背后的哲学思想。无论你是开发者、系统管理员还是普通用户,了解这些基础知识都将有助于你更好地利用Linux的强大功能。 ####
39 1
|
2月前
|
Linux Shell
Linux系统编程:掌握popen函数的使用
记得在使用完 `popen`打开的流后,总是使用 `pclose`来正确关闭它,并回收资源。这种做法符合良好的编程习惯,有助于保持程序的健壮性和稳定性。
111 6
|
2月前
|
Linux Shell
Linux系统编程:掌握popen函数的使用
记得在使用完 `popen`打开的流后,总是使用 `pclose`来正确关闭它,并回收资源。这种做法符合良好的编程习惯,有助于保持程序的健壮性和稳定性。
144 3
|
2月前
|
Linux
在Linux内核中根据函数指针输出函数名称
在Linux内核中根据函数指针输出函数名称
|
3月前
|
项目管理 敏捷开发 开发框架
敏捷与瀑布的对决:解析Xamarin项目管理中如何运用敏捷方法提升开发效率并应对市场变化
【8月更文挑战第31天】在数字化时代,项目管理对软件开发至关重要,尤其是在跨平台框架 Xamarin 中。本文《Xamarin 项目管理:敏捷方法的应用》通过对比传统瀑布方法与敏捷方法,揭示敏捷在 Xamarin 项目中的优势。瀑布方法按线性顺序推进,适用于需求固定的小型项目;而敏捷方法如 Scrum 则强调迭代和增量开发,更适合需求多变、竞争激烈的环境。通过详细分析两种方法在 Xamarin 项目中的实际应用,本文展示了敏捷方法如何提高灵活性、适应性和开发效率,使其成为 Xamarin 项目成功的利器。
53 1
|
3月前
|
Linux
揭秘Linux心脏:那些让你的编程事半功倍的主要系统调用
【8月更文挑战第31天】Linux中的系统调用是操作系统提供给应用程序的接口,用于请求内核服务,如文件操作、进程控制等。本文列举了22种主要系统调用,包括fork()、exec()、exit()、wait()、open()、close()、read()、write()等,并通过示例代码展示了如何使用fork()创建新进程及使用open()、write()、close()操作文件。这些系统调用是Linux中最基本的接口,帮助应用程序与内核交互。
52 1
|
3月前
|
Linux PHP
Linux CentOS 宝塔 Suhosin禁用php5.6版本eval函数详细图文教程
【8月更文挑战第27天】本文介绍两种禁用PHP执行的方法:使用`PHP_diseval_extension`禁用和通过`suhosin`禁用。由于`suhosin`不支持PHP8,仅适用于PHP7及以下版本,若服务器安装了PHP5.6,则需对应安装`suhosin-0.9.38`版本。文章提供了详细的安装步骤,并强调了宝塔环境下与普通环境下的PHP路径差异。安装完成后,在`php.ini`中添加`suhosin.so`扩展并设置`executor.disable_eval = on`以禁用执行功能。最后通过测试代码验证是否成功禁用,并重启`php-fpm`服务生效。
49 2
|
3月前
|
C语言
Linux0.11 系统调用进程创建与执行(九)(下)
Linux0.11 系统调用进程创建与执行(九)
34 1
|
3月前
|
存储 Linux 索引
Linux0.11 系统调用进程创建与执行(九)(上)
Linux0.11 系统调用进程创建与执行(九)
78 1
下一篇
无影云桌面