每天进步一点点——Linux中的文件描写叙述符与打开文件之间的关系

转载请说明出处:http://blog.csdn.net/cywosp/article/details/38965239

1. 概述

在Linux系统中一切皆能够看成是文件,文件又可分为:普通文件、文件夹文件、链接文件和设备文件。文件描写叙述符(file descriptor)是内核为了高效管理已被打开的文件所创建的索引。其是一个非负整数(一般是小整数),用于指代被打开的文件。全部运行I/O操作的系统调用都通过文件描写叙述符。程序刚刚启动的时候。0是标准输入,1是标准输出,2是标准错误。

假设此时去打开一个新的文件,它的文件描写叙述符会是3。

POSIX标准要求每次打开文件时(含socket)必须使用当前进程中最小可用的文件描写叙述符号码,因此,在网络通信过程中稍不注意就有可能造成串话。标准文件描写叙述符图例如以下:

文件描写叙述与打开的文件相应模型例如以下图:

2. 文件描写叙述限制

在编写文件操作的或者网络通信的软件时,刚開始学习的人一般可能会遇到“Too many open files”的问题。这主要是由于文件描写叙述符是系统的一个重要资源。尽管说系统内存有多少就能够打开多少的文件描写叙述符。可是在实际实现过程中内核是会做对应的处理的,一般最大打开文件数会是系统内存的10%(以KB来计算)(称之为系统级限制)。查看系统级别的最大打开文件数能够使用sysctl
-a | grep fs.file-max命令查看。

与此同一时候,内核为了不让某一个进程消耗掉全部的文件资源。其也会对单个进程最大打开文件数做默认值处理(称之为用户级限制),默认值通常是1024,使用ulimit -n命令能够查看。在Webserver中,通过更改系统默认值文件描写叙述符的最大值来优化server是最常见的方式之中的一个。详细优化方式请查看http://blog.csdn.net/kumu_linux/article/details/7877770

3. 文件描写叙述符合打开文件之间的关系

每个文件描写叙述符会与一个打开文件相相应。同一时候。不同的文件描写叙述符也会指向同一个文件。同样的文件能够被不同的进程打开也能够在同一个进程中被多次打开。系统为每个进程维护了一个文件描写叙述符表,该表的值都是从0開始的。所以在不同的进程中你会看到同样的文件描写叙述符,这样的情况下同样文件描写叙述符有可能指向同一个文件,也有可能指向不同的文件。

详细情况要详细分析。要理解详细其概况怎样,须要查看由内核维护的3个数据结构。

1. 进程级的文件描写叙述符表

2. 系统级的打开文件描写叙述符表

3. 文件系统的i-node表

进程级的描写叙述符表的每一条目记录了单个文件描写叙述符的相关信息。

1. 控制文件描写叙述符操作的一组标志。

(眼下,此类标志仅定义了一个,即close-on-exec标志)

2. 对打开文件句柄的引用

内核对所有打开的文件的文件维护有一个系统级的描写叙述符表格(open file description table)。有时,也称之为打开文件表(open file table)。并将表格中各条目称为打开文件句柄(open file handle)。

一个打开文件句柄存储了与一个打开文件相关的所有信息,例如以下所看到的:

1. 当前文件偏移量(调用read()和write()时更新,或使用lseek()直接改动)

2. 打开文件时所使用的状态标识(即,open()的flags參数)

3. 文件訪问模式(如调用open()时所设置的仅仅读模式、仅仅写模式或读写模式)

4. 与信号驱动相关的设置

5. 对该文件i-node对象的引用

6. 文件类型(比如:常规文件、套接字或FIFO)和訪问权限

7. 一个指针,指向该文件所持有的锁列表

8. 文件的各种属性,包含文件大小以及与不同类型操作相关的时间戳

下图展示了文件描写叙述符、打开的文件句柄以及i-node之间的关系。图中。两个进程拥有诸多打开的文件描写叙述符。

在进程A中。文件描写叙述符1和30都指向了同一个打开的文件句柄(标号23)。这可能是通过调用dup()、dup2()、fcntl()或者对同一个文件多次调用了open()函数而形成的。

进程A的文件描写叙述符2和进程B的文件描写叙述符2都指向了同一个打开的文件句柄(标号73)。这样的情形可能是在调用fork()后出现的(即,进程A、B是父子进程关系)。或者当某进程通过UNIX域套接字将一个打开的文件描写叙述符传递给还有一个进程时,也会发生。

再者是不同的进程独自去调用open函数打开了同一个文件,此时进程内部的描写叙述符正好分配到与其它进程打开该文件的描写叙述符一样。

此外,进程A的描写叙述符0和进程B的描写叙述符3分别指向不同的打开文件句柄。但这些句柄均指向i-node表的同样条目(1976),换言之,指向同一个文件。发生这样的情况是由于每一个进程各自对同一个文件发起了open()调用。同一个进程两次打开同一个文件。也会发生类似情况。

4. 总结

1. 因为进程级文件描写叙述符表的存在,不同的进程中会出现同样的文件描写叙述符。它们可能指向同一个文件,也可能指向不同的文件

2. 两个不同的文件描写叙述符,若指向同一个打开文件句柄,将共享同一文件偏移量。因此,假设通过当中一个文件描写叙述符来改动文件偏移量(由调用read()、write()或lseek()所致),那么从还有一个描写叙述符中也会观察到变化,不管这两个文件描写叙述符是否属于不同进程,还是同一个进程。情况都是如此。

3. 要获取和改动打开的文件标志(比如:O_APPEND、O_NONBLOCK和O_ASYNC),可运行fcntl()的F_GETFL和F_SETFL操作,其对作用域的约束与上一条颇为类似。

4. 文件描写叙述符标志(即。close-on-exec)为进程和文件描写叙述符所私有。对这一标志的改动将不会影响同一进程或不同进程中的其它文件描写叙述符

參考

[1] http://blog.chinaunix.net/uid-20633888-id-2747146.html

[2] http://www.cppblog.com/guojingjia2006/archive/2012/11/21/195450.html

[3] http://blog.csdn.net/kumu_linux/article/details/7877770

[4] 《Linux/UNIX系统编程手冊》

时间: 2024-11-12 01:15:14

每天进步一点点——Linux中的文件描写叙述符与打开文件之间的关系的相关文章

Linux 文件描写叙述符设置为非堵塞的方法

通过fcntl设置文件描写叙述符属性 fcntl即F_SETFL,F_GETFL的使用,设置文件的flags,堵塞设置成非堵塞,非堵塞设置成堵塞(这连个在server开发中能够封装为基本函数) 1.获取文件的flags,即open函数的第二个參数: flags = fcntl(fd,F_GETFL,0); 2.设置文件的flags: fcntl(fd,F_SETFL,flags); 3.添加文件的某个flags.比方文件是堵塞的,想设置成非堵塞: flags = fcntl(fd,F_GETFL

每天进步一点点——Linux中的线程局部存储(一)

转载请说明出处:http://blog.csdn.net/cywosp/article/details/26469435 在Linux系统中使用C/C++进行多线程编程时,我们遇到最多的就是对同一变量的多线程读写问题,大多情况下遇到这类问题都是通过锁机制来处理,但这对程序的性能带来了很大的影响,当然对于那些系统原生支持原子操作的数据类型来说,我们可以使用原子操作来处理,这能对程序的性能会得到一定的提高.那么对于那些系统不支持原子操作的自定义数据类型,在不使用锁的情况下如何做到线程安全呢?本文将从

Linux中的默认权限与隐藏权限(文件、目录)

一个文件(或目录)拥有若干个属性,包括(r/w/x)等基本属性,以及是否为目录(d)与文件(-)或连接文件(l)等属性.此外,Linux还可以设置其他系统安全属性,使用chattr来设置,以lsattr来查看,最重要的是可以设置其不可修改的特性,即便是文件的拥有者都不能进行修改.这个属性相当重要,尤其是在安全机制方面(security). 文件默认权限:umask 当建立一个新的文件或目录时,它的默认属性是与umask有关的.通常,umask就是指定当前用户在建立文件或目录时的属性默认值.那么,

Linux中文件描述符和打开文件之间的关系

Linux中文件描述符和打开文件之间的关系 文件描述符: 在形式上是一个非负整数.实际上,它是一个索引值,指向内核为每一个进程所维护的该进程打开文件的记录表. Linux中的文件类型 Linux系统中把一切都看做文件,包括普通文件-.目录文件d.字符设备文件c.块设备文件b.符号链接文件l.文件描述符是内核为了高效管理已被打开的文件所创建的索引(一个非负整数),用于指代已被打开的文件,Linux下所有的的I/O操作的系统调用都是通过文件描述符执行.例如0表示标准输入.1表示标准输出.3表示标准错

Linux中的元字符和转义符 单引号 硬引号 双引号 软引号

Linux中的元字符和转义符  单引号  硬引号  双引号  软引号 Linux就这个范儿 Linux就这个范儿 P182单引号:硬引号,所有元字符特殊意义都会关掉双引号:软引号,只允许出现特定元字符,对特殊字符会进行解析 元字符

Eclipse中的一个小插件easyExplorer--用于打开文件所在目录

最近格式化了硬盘,里面的诸多有价值的插件都没了,开发时需要打开java文件所在的目录,每次都很麻烦,实在无法忍受,就又重新下载了一个小插件,用以很方便的打开文件所在目录: easyExplorer 仅有7kb 一:官方下载地址: http://sourceforge.net/projects/easystruts/ 由于天朝网络问题,等个大概十来秒的时间出现下图: 二:配置方式: 直接将下载后的文件:org.sf.easyexplore_1.0.4.jar复制到eclipse下的plugins下

Android中调用系统所装的软件打开文件(转)

Android中调用系统所装的软件打开文件(转) 在应用中如何调用系统所装的软件打开一个文件,这是我们经常碰到的问题,下面是我所用到的一种方法,和大家一起分享一下! 这个是打开文件的一个方法: Java代码 /** * 打开文件 * @param file */ private void openFile(File file){ Intent intent = new Intent(); intent.addFlags(Intent.FLAG_ACTIVITY_NEW_TASK); //设置in

Unix系统编程()文件描述符和打开文件之间的关系

目前学习到的是一个文件描述符对应着一个打开的文件,似乎是一一对应的关系.但是实际上并不是这样的.多个文件描述符指向同一个打开的文件,是可能的也是必要的.这些文件描述符可以在相同或者不同的进程中打开. 要理解具体情况,需要查看内核维护的3个数据结构. 进程级的文件描述符表 系统级的打开文件表 文件系统的i-node表 针对每个进程,内核为其维护打开文件的描述符(open file descriptor)表.该表的每一条目都记录了单个文件描述符的相关信息.包括有一下信息: 控制文件描述符操作的一组标

android 中View, Window, Activity, WindowManager,ViewRoot几者之间的关系

(1)View:最基本的UI组件,表示屏幕上的一个矩形区域. (2)Window: 表示一个窗口,不一定有屏幕那么大,可以很大也可以很小: 它包含一个View tree和窗口的layout 参数. View tree的root View可以通过getDecorView得到.还可以设置Window的Content View. (3)Activity:Activity包含一个Window,该Window在Activity的attach方法中通过调用 PolicyManager.makeNewWind