linux文件系统系统调用实测的一个方法

简介:


通过几个C代码来检测一下文件系统相关系统调用时间,以及和标准I/O库函数的性能差异。主要以read和fread函数为例。

1.1.1 read系统调用开销

我们来模拟一下read系统调用的开销情况,

代码如下:

#include <unistd.h>

#include <sys/stat.h>

#include <fcntl.h>

#include <stdlib.h>

 

int main()

{

      char c;

      int in,out;

      in = open("file.in",O_RDONLY);

      out=open("file.out",O_WRONLY|O_CREAT,S_IRUSR|S_IWUSR);

      while(read(in,&c,1)==1)

           write(out,&c,1);

      exit(0);

}

创建输入文件file.in如下:

dd if=/dev/zero of=file.in bs=1 count=1024000

执行测试如下:

# time ./a.out

real    0m1.080s

user    0m0.064s

sys   0m1.012s

发现主要时间是花在了sys上的,即内核态,当然我们知道是系统调用了,因为就是我们写的嘛。

       可以使用strace来跟踪程序,就会不断显示系统调用write和read了。

       通过测试时间,其实我们计算得到一次write+read的时间的。就是1.012s/1024000=0.98us。

       修改程序如下:

#include <unistd.h>

#include <sys/stat.h>

#include <fcntl.h>

#include <stdlib.h>

 

int main()

{

      char c;

      int in,out;

      in = open("file.in",O_RDONLY);

      while(read(in,&c,1)==1)

           ;

      exit(0);

}

       程序中,去掉了write只留下了read系统调用,运行后时间如下:

# time ./a.out

real    0m0.222s

user    0m0.012s

sys   0m0.208s

可以计算得到每次read的大概时间为0.208/1024000=0.2us.在不同的机器上运行会有不同的结果,大家可以自行测试,祝玩的愉快。

后续会将该程序进行优化。

1.1.2 read系统调用开销二

将代码进行优化如下:

#include <unistd.h>

#include <sys/stat.h>

#include <fcntl.h>

#include <stdlib.h>

 

int

main ()

{

  char block[1024];

  char c;

  int in, out;

  in = open ("file.in", O_RDONLY);

  out = open ("file.out", O_WRONLY | O_CREAT, S_IRUSR | S_IWUSR);

  while (read (in, block, sizeof(block)) >0 )

    write (out, &c, 1);

  exit (0);

}

       一次read的字节数量修改为1024个字节。

执行后如下,

# time ./a.out

real    0m0.002s

user    0m0.000s

sys 0m0.000s

发现执行时间大幅降低了,性能得到了优化,理论上应该快了千倍被,毕竟系统调用次数减少了千次。

1.1.3 标准I/O库

标准I/O库是由stdio及头文件stdio.h为底层i/o系统调用提供的一个通用接口,例如fopen,fread,fclose等。现在是ANSI标准C的一部分。

1.1.4 性能差异

使用标准I/O库的函数来进行测试,代码如下,

#include <stdio.h>

#include <stdlib.h>

int

main ()

{

      int c;

      FILE  *in,*out;

 

     

  in = fopen ("file.in", "r");

  out = fopen ("file.out", "w");

  while((c=fgetc(in))!=EOF)

        fputc(c,out);

  exit (0);

}

编译执行,

# time ./a.out

real    0m0.036s

user    0m0.032s

sys   0m0.000s

       发现sys时间几乎为0,说明主要是在用户态的操作,相比read系统调用所化时间(如下)快了很多。

real    0m1.080s

user    0m0.064s

sys   0m1.012s

       这个主要是因为在 stdio 库在 FILE 结构里使用了一个内部缓冲区。只有在缓冲区满时候才进行底层系统调用再刷数据,当然比直接系统调用快很多了。
目录
相关文章
|
1月前
|
Ubuntu Linux 网络安全
在Linux上安装软件有多种方法
在Linux上安装软件有多种方法
100 64
|
2天前
|
存储 Linux 文件存储
Linux文件系统
Linux文件系统 一切皆文件 在Linux中,“一切皆文件”的概念意味着系统中的所有资源,包括硬件设备、目录及进程等,均被视为文件。这种设计简化了操作和管理,具体包括: 普通文件:存储数据的常规文件。 目录文件:包含其他文件和子目录的文件。 进程文件:在/proc目录下代表系统中运行的进程。 设备文件:位于/dev目录,代表硬件设备。 网络字节流套接字文件:用于网络通信的数据流。 链接文件:指向另一个文件的符号链接或硬链接。 管道文件:用于进程间通信的文件。
25 7
|
8天前
|
Linux Shell 数据库
文件查找是Linux用户日常工作的重要技能介绍了几种不常见的文件查找方法
文件查找是Linux用户日常工作的重要技能。本文介绍了几种不常见的文件查找方法,包括使用`find`和`column`组合、`locate`和`mlocate`快速查找、编写Shell脚本、使用现代工具`fd`、结合`grep`搜索文件内容,以及图形界面工具如`Gnome Search Tool`和`Albert`。这些方法能显著提升文件查找的效率和准确性。
28 2
|
18天前
|
网络协议 Linux 调度
深入探索Linux操作系统的心脏:内核与系统调用####
本文旨在揭开Linux操作系统中最为核心的部分——内核与系统调用的神秘面纱,通过生动形象的语言和比喻,让读者仿佛踏上了一段奇妙的旅程,从宏观到微观,逐步深入了解这两个关键组件如何协同工作,支撑起整个操作系统的运行。不同于传统的技术解析,本文将以故事化的方式,带领读者领略Linux内核的精妙设计与系统调用的魅力所在,即便是对技术细节不甚了解的读者也能轻松享受这次知识之旅。 ####
|
14天前
|
运维 Linux
Linux查找占用的端口,并杀死进程的简单方法
通过上述步骤和命令,您能够迅速识别并根据实际情况管理Linux系统中占用特定端口的进程。为了获得更全面的服务器管理技巧和解决方案,提供了丰富的资源和专业服务,是您提升运维技能的理想选择。
16 1
|
14天前
|
运维 安全 Linux
Linux文件清空的五种方法总结分享
每种方法各有优势,选择最合适的一种或几种,可以极大提高您的工作效率。更多有关Linux系统管理的技巧与资源,欢迎访问,持续提升您的运维技能。
60 1
|
15天前
|
缓存 算法 安全
深入理解Linux操作系统的心脏:内核与系统调用####
【10月更文挑战第20天】 本文将带你探索Linux操作系统的核心——其强大的内核和高效的系统调用机制。通过深入浅出的解释,我们将揭示这些技术是如何协同工作以支撑起整个系统的运行,同时也会触及一些常见的误解和背后的哲学思想。无论你是开发者、系统管理员还是普通用户,了解这些基础知识都将有助于你更好地利用Linux的强大功能。 ####
25 1
|
1月前
|
Linux 数据安全/隐私保护 索引
linux inode索引节点使用率100% 解决+hustoj忘记密码+最新MDK注册方法
linux inode索引节点使用率100% 解决+hustoj忘记密码+最新MDK注册方法
40 1
|
1月前
|
监控 安全 Linux
使用NRPE和Nagios监控Linux系统资源的方法
通过遵循以上步骤,可以有效地使用NRPE和Nagios监控Linux系统资源,确保系统运行稳定,并及时响应任何潜在的问题。这种方法提供了高度的可定制性和灵活性,适用于从小型环境到大型分布式系统的各种监控需求。
42 2
|
2月前
|
Shell Linux Python
python执行linux系统命令的几种方法(python3经典编程案例)
文章介绍了多种使用Python执行Linux系统命令的方法,包括使用os模块的不同函数以及subprocess模块来调用shell命令并处理其输出。
29 0