原文链接:https://blog.csdn.net/lf_2016/article/details/54587020

一、linux结构图简要分析 
这里写图片描述

典型的linux结构如图: 
操作系统与内核基本上是相同的,只不过操作系统在内核的基础上还有一些延伸,包括了提供基础服务的组件。操作系统对下通过驱动程序管理硬件。对上的话则暴露出来一些接口,供上层调用,这些接口就叫做系统调用。shell是包裹在linux内核外层的,一个可通过一系列的linux命令对操作系统发出相关指令的人机交互界面。库函数是在系统调用的基础上经过扩展和封装。 
可以看到,有两种调用函数的方式。一种是系统调用,系统调用是操作系统的一部分 ,它在内核空间中执行。另一种是库调用,函数库调用是语言或应用程序的一部分,它在用户空间中执行。

二、系统调用和库函数调用 
下面来看看这两种方式有什么不同: 
这里写图片描述
实际上,库函数是对系统调用的一层封装,因此在用库函数对文件操作的时候,必然会引起系统调用。也就是说,库函数调用实际上是通过系统调用实现的,例如:C库函数fwrite()就是通过write实现的。 
但是库函数在调用的时候有函数调用开销,而系统调用比库函数调用还要慢的多,因为它需要把上下文环境切换到内核模式。 
系统调用通常是用于底层文件的访问,例如在驱动程序中对设备文件的直接访问。系统调用与操作系统是相关的,因此一般没有跨操作系统的可移植性。系统调用发生在内核空间,因此如果在用户空间的一般应用程序中使用系统调用来进行文件操作的话,会有从用户空间切换到内核空间的开销。

系统调用: 
系统调用提供的函数如open,close,read,write等,需要包含unistd.h头文件。以write为例:它的函数原型是size_t write(int fd,const void *buf,size_t nbytes),其操作对象为文件描述符fd(file descriptor),要写一个文件,必须先以open系统调用打开一个文件,获得所打开文件的文件描述符fd。文件描述符fd是一个整数,它的取值从0开始。linux系统默认分配了3个文件描述符值:0对应stdin(标准输入),1对应stdout(标准输出),2对应stderr(标准错误)。每当打开一个新的文件,就会分配一个fd,这个fd的取值是当前未分配的文件描述符值的最小值。 
例:现在0,1,2,3,4已经被分配,然后3又被关闭,这时候再打开一个文件的话,就会把3分配出去。

库函数调用: 
标准C库函数如fopen,fread,fwrite等,需要包含stdio.h头文件。库函数是对系统调用的一层封装,底层还是通过系统调用实现的。每打开一个文件所获得的文件指针FILE都有一个内核空间的文件描述符fd与之对应

问题:既然使用库函数调用也有系统调用的开销,那为什么不直接使用系统调用呢??? 
这是因为,读写文件通常是大量的数据(这种大量是相对于底层驱动的系统调用所实现的数据操作单位而言),使用库函数调用可以大大减少系统调用的次数。这是因为缓冲区技术。在用户空间和内核空间,对文件操作都使用了缓冲区,当内核缓冲区写满之后或写结束之后才将内核缓冲区内容写到文件对应的硬件媒介中

 

 

 

一:系统调用和库函数调用的区别:

1:系统调用是最底层的应用,是面向硬件的。而库函数的调用是面向开发的,相当于应用程序的API(即预先定义好的函数)接口;

2:各个操作系统的系统调用是不同的,因此系统调用一般是没有跨操作系统的可移植性,而库函数的移植性良好(c库在Windows和Linux环境下都可以操作);

3:库函数属于过程调用,调用开销小;系统调用需要在用户空间和内核上下文环境切换,开销较大;

4:库函数调用函数库中的一段程序,这段程序最终还是通过系统调用来实现的;系统调用调用的是系统内核的服务。

二:具体的执行过程
系统调用是发生在内核空间的,所以如果在用户空间使用系统调用进行文件操作,必然会引起用户空间到内核空间切换的开销。( 系统调用是一个很费时的操作)事实上,  即使在用户空间使用库函数对文件进行操作,因为文件总是存在于存储介质上,因此,读写操作都是对硬件(存储器)的操作,所以肯定会引起系统调用,也就是说,库函数对文件的操作是通过系统调用来实现的,  (即库函数封装了系统调用的很多细节)。例如C库函数的fopen()就是通过系统调用open ()来实现的。

三:即使使用库函数也会有系统调用的开销,为什么不直接使用系统调用呢?

这是因为,文件的读写操作通常是大量的数据(大量是底层实现而言),这时,使用库函数可以大大减少系统调用的次数。这一结果源于缓冲区技术,在内核空间和用户空间,对文件操作都使用了缓冲区,例如用fwrite()写文件,都是先将内容写到用户空间缓冲区,当用户空间缓冲去写满或者写操作结束时,才将用户缓冲区的内容写到内核缓冲区,同样的道理,当内核缓冲区写满或者写结束时才将内核缓冲区的内容写到文件对应的硬件媒介上

posted on 2018-09-17 17:00  jasmine_qy  阅读(1075)  评论(0编辑  收藏  举报