引文

在 Linux 的世界里,一切设备皆文件。对文件的操作都是通过文件描述符(fd)来进行的。

Linux 中有7种文件类型:

文件类型 文件类型描述 符号
普通文件 最常使用的一类文件,其特点是不包含有文件系统信息的结构信息。这种类型的文件是按照其内部结构又可分为纯文本文件(ASCII)、二进制文件(binary)、数据格式的文件(data)、各种压缩文件。 REG (-)
目录文件 用于存放文件名以及其相关信息的文件,是内核组织文件系统的基本节点。 DIR (d)
块设备 块设备文件 : 就是存储数据以供系统存取的接口设备,简单而言就是硬盘。 BLK (b)
字符设备 字符设备文件:即串行端口的接口设备,例如键盘、鼠标等等。 CHR (c)
套接字 这类文件通常用在网络数据连接。可以启动一个程序来监听客户端的要求,客户端就可以通过套接字来进行数据通信。 SOCK(s)
管道 一种很特殊的文件,主要用于不同进程的信息传递。 FIFO (p)
链接 一种特殊文件,指向一个真实存在的文件链接,类似于Windows下的快捷方式,链接文件的不同,又可分为硬链接和软链接。 LNK (l)
未知

我们可以通过 Il 命令来查看文件类型:

1. 文件描述符

**文件描述符相当于非负的索引。**我们可以通过文件描述符操作相应的文件。

Linux进程会默认打开三个文件描述符:

  • 0 :标准输入 stdin
  • 1 :标准输出 stdout
  • 2 :标准错误 stderror

文件描述符的分配原则:分配出未被占用的最小文件描述符。

一般来说,我们打开文件后占用一个文件描述符3(0、1、2已被标准文件描述符占用),下一个则是4(如果它没被占用的话)…

#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <stdio.h>
#include <unistd.h>int main() {// int open(const char *pathname, int flags);//     参数://         - pathname:要打开的文件路径//         - flags:对文件的操作权限设置还有其他的设置//           O_RDONLY,  O_WRONLY,  O_RDWR  这三个设置是互斥的//     返回值:返回一个新的文件描述符,如果调用失败,返回-1// 打开或创建一个文件int fd1 = open("temp.txt", O_RDONLY | O_CREAT);if(fd1 == -1) {perror("open");return -1;}int fd2 = open("temp.txt", O_RDONLY );if(fd2 == -1) {perror("open");return -1;}printf("fd1: %d\n", fd1);printf("fd2: %d\n", fd2);// 关闭文件描述符close(fd1);close(fd2);return 0;
}

运行结果:

我们观察上述代码可以发现:同一个进程的不同文件描述符可以指向同一个文件。

同时,当我们复制该会话(相当于打开了一个新进程),再运行该程序:

发现fd1 = 3 和 fd2 = 4,这表明:不同进程可以拥有相同的文件描述符。

进而,不同进程的不同文件描述符也可以指向同一个文件。

2. dup 和 dup2

2.1 dup函数

Linux下可通过 man 命令查看它们的具体使用。

ydh@ubuntu:~/linux/test_dup$ man 2 dup
ydh@ubuntu:~/linux/test_dup$ man 2 dup

头文件:#include <unistd.h>

函数原型:int dup(int oldfd);

作用:复制一个新的文件描述符,从空闲的文件描述符表中找一个最小的,作为新的拷贝的文件描述符。

#include <unistd.h>
#include <stdio.h>
#include <fcntl.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <string.h>int main() {int fd = open("a.txt", O_RDWR | O_CREAT, 0664);int fd1 = dup(fd);if(fd1 == -1) {perror("dup");return -1;}printf("fd : %d , fd1 : %d\n", fd, fd1);close(fd);char * str = "hello,world\n";int ret = write(fd1, str, strlen(str));if(ret == -1) {perror("write");return -1;}close(fd1);return 0;
}

运行结果:

观察运行结果可以发现,fd1被分配到了一个最小的空闲文件描述符;fd1 和 fd 指向同一个文件,当我们通过操作fd1进行写数据时,fd指向的文件发生了修改。

2.2 dup2函数

#include <unistd.h>

int dup2(int oldfd, int newfd);

作用:重定向文件描述符

假如 oldfd 指向 a.txt, 而 newfd 指向 b.txt,调用函数成功后:newfd 和 b.txt 做close, newfd 指向了 a.txt

注:oldfd 必须是一个有效的文件描述符,oldfd和newfd值相同,相当于什么都没有做

#include <unistd.h>
#include <stdio.h>
#include <string.h>
#include <sys/stat.h>
#include <sys/types.h>
#include <fcntl.h>int main() {int fd = open("1.txt", O_RDWR | O_CREAT, 0664);if(fd == -1) {perror("open");return -1;}int fd1 = open("2.txt", O_RDWR | O_CREAT, 0664);if(fd1 == -1) {perror("open");return -1;}printf("fd : %d, fd1 : %d\n", fd, fd1);int fd2 = dup2(fd, fd1);if(fd2 == -1) {perror("dup2");return -1;}// 通过fd1去写数据,实际操作的是1.txt,而不是2.txtchar * str = "hello, dup2";int len = write(fd1, str, strlen(str));if(len == -1) {perror("write");return -1;}printf("fd : %d, fd1 : %d, fd2 : %d\n", fd, fd1, fd2);close(fd);close(fd1);return 0;
}

运行结果:

上述代码运行成功后,fd2从原来的2.txt,指向了1.txt。

3.fcntl函数

fcntl 函数所需头文件:

#include <unistd.h>
#include <fcntl.h>

函数原型:

int fcntl(int fd, int cmd, ... /* arg */ );

参数含义:

​ fd : 表示需要操作的文件描述符

​ cmd: 表示对文件描述符进行如何操作

​ … /* arg */:为可选参数

fcntl的两个常用功能:

  1. 复制文件描述符
 int fd = open("1.txt", O_RDONLY);// F_DUPFD : 复制文件描述符,复制的是第一个参数fd,得到一个新的文件描述符(返回值)int ret = fcntl(fd, F_DUPFD);
  1. 修改或者获取文件描述符状态flag
#include <unistd.h>
#include <fcntl.h>
#include <stdio.h>
#include <string.h>int main() {// 2.修改或者获取文件描述符状态flagint fd = open("1.txt", O_RDWR);if(fd == -1) {perror("open");return -1;}// 获取文件描述符状态flagint flag = fcntl(fd, F_GETFL);if(flag == -1) {perror("fcntl");return -1;}flag |= O_APPEND;   // flag = flag | O_APPEND// 修改文件描述符状态的flag,给flag加入O_APPEND这个标记int ret = fcntl(fd, F_SETFL, flag);if(ret == -1) {perror("fcntl");return -1;}char * str = "world";write(fd, str, strlen(str));close(fd);return 0;
}

上述代码为文件描述符增加了O_APPEND(追加)选项,向1.txt中追加了"world"。

其他更多选项可通过man指向进行查询。

ydh@ubuntu:~/linux/test_fcntl$ man 2 fcntl

Linux下的文件描述符相关推荐

  1. Linux下利用文件描述符恢复的成功失败实验

    数据误删除是作为初级运维人员常常遇到的"低级错误",一些有经验的老手有时也在疲劳.不冷静的情况下"马失前蹄".一旦误删除数据文件,尽快采用影响最小.最迅速的手段 ...

  2. linux 下修改文件描述符限制

    修改linux的最大文件句柄数限制 对于一般的应用来说(象Apache.系统进程)1024完全足够使用.但是如何象squid.mysql.java等单进程处理大量请求的应用来说就有点捉襟见肘了.如果单 ...

  3. Linux中的文件描述符与打开文件之间的关系

    1. 概述 在Linux系统中一切皆可以看成是文件,文件又可分为:普通文件.目录文件.链接文件和设备文件.文件描述符(file descriptor)是内核为了高效管理已被打开的文件所创建的索引,其是 ...

  4. Linux C:文件描述符、IO重定向、恢复标准输入输出

    目录 一.文件描述符 二.IO重定向 三.重定向回终端.伪终端 四.恢复标准输入输出 一.文件描述符 在Linux中,文件描述符是一个非负整数的数据类型.是FILE结构体中的一个成员属性. 每打开或者 ...

  5. linux文件描述符有什么用,linux上的文件描述符3有什么特别之处?

    我的工作,那将在Linux和Mac OS X上运行的服务器应用程序它是这样的:linux上的文件描述符3有什么特别之处? 启动主要应用 控制器进程的叉 调用lock_down()在控制过程中 再次叉终 ...

  6. Linux中对文件描述符的操作(FD_ZERO、FD_SET、FD_CLR、FD_ISSET

    在Linux中,内核利用文件描述符(File Descriptor)即文件句柄,来访问文件.文件描述符是非负整数.打开现存文件或新建文件时,内核会返回一个文件描述符.读写文件也需要使用文件描述符来指定 ...

  7. Linux网络编程--文件描述符

    文件描述符 在Unix和Unix-like操作系统中,文件描述符(file descriptor, FD)是一个文件或者像pipe或者network socket等之类的输入/输出源的唯一标识. 文件 ...

  8. linux exec操作文件描述符

    linux每一个打开文件都会关联一个文件描述符,需要的时候我们可以使用exec命令指定一个大于3的数字作为文件 linux默认文件描述符 每打开一个shell就会打开默认的三个文件描述符描0,1,2, ...

  9. linux 标准输出 复制,使用LINUX dup2 复制文件描述符到标准输出STDOUT_FILENO

    7 8 #include 9 #include 10 #include 11 #include 12 #include 13 #include 14 15 16 17 int main(int arg ...

最新文章

  1. 如何保证工业相机工作的精准与稳定?
  2. 汇总 | OpenCV4中的非典型深度学习模型
  3. 谁是真正的深度学习?英特尔高管与AI大神再“论剑”
  4. 终结 Python 原生字典?这个库要逆天改命了
  5. 让vue-router渲染为指定的标签
  6. asp:HyperLink中 Eval要用string.Format绑定
  7. jQuery系列(十四):jQuery中的ajax
  8. php邮件发送tp,Thinkphp5 邮件发送Thinkphp发送邮件
  9. leetcode586. 订单最多的客户(SQL)
  10. python 释放链表节点_redis:链表
  11. 正则表达式--内功心法
  12. Delphi Access violations 问题的解决之道[转]
  13. 如何让自己的CS水平更进一步?(二)了解武器
  14. AutoCAD-图纸集使用方法
  15. 计数器verilog代码(quartus II)
  16. Youtube更改视频原始语言
  17. 【算法学习笔记】09.数据结构基础 二叉树初步练习2
  18. 一个基于server酱推送服务的laravel package
  19. HTML学习笔记——框架结构
  20. 网络渗透——CTF实践

热门文章

  1. 头文件为什么要加#ifndef #define #endif
  2. 顺序表的建立,插入,删除,二分(折半)查找,监视哨查找,冒泡排序,选择排序,直接插入排序
  3. IFDEF和ENDIF的理解
  4. 羊了个羊爆火的模式分析
  5. 彩色图文验证码(英文、数字、中文)
  6. ViewBag的使用方法
  7. Python实用记录(一):如何将不同类型视频按关键帧提取并保存图片,实现图片裁剪功能
  8. 下次总能很轻松找到位置皮鞋
  9. iOS线上APP崩溃(Crash)分析
  10. 使用Parallels Desktop centos7设置静态ip