当前位置: 首页 > news >正文

Linux操作系统3-文件与IO操作2(文件描述符fd与文件重定向)

上篇文章:Linux操作系统3-文件与IO操作1(从C语言IO操作到系统调用)-CSDN博客

本篇代码Gitee仓库:myLerningCode · 橘子真甜/Linux操作系统与网络编程学习 - 码云 - 开源中国 (gitee.com)

本篇重点:文件描述符fd与文件重定向

目录

一. 文件描述符fd及其分配规则

二. 文件重定向

2.1 ">"  ">>"  "<"  命令

 2.2 重定向的本质⭐

2.3 使用dup2完成重定向 ⭐

a dup2完成输出重定向

b dup2完成追加重定向

c dup2输入重定向 

​编辑 三. 如何理解Linux下一切皆文件?

四. 下篇内容:C语言FILE与用户级缓冲区与文件系统


一. 文件描述符fd及其分配规则

        在上一篇文章中,我们使用open系统调用打开文件之后。看到返回的fd是一个数字,并且我们输出了stdin,stdout,stderror的文件fd。发现它们分别是1,2,3。

        当时的测试代码如下:

#include <iostream>#include <cstdio>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>#define MY_FILE "log.txt"int main()
{umask(0);int fd1 = open(MY_FILE, O_RDWR | O_CREAT | O_TRUNC, 0666);printf("stdin fd -> %d\n", stdin->_fileno);printf("stdout fd -> %d\n", stdout->_fileno);printf("stderr fd -> %d\n", stderr->_fileno);printf("log.txt fd -> %d\n", fd1);return 0;
}

测试结果如下:

如果我们关闭了stderr在打开log.txt的话,log.txt的文件fd是不是就是2?

测试代码 (注意不能直接使用系统调用close关闭FILE结构体的_fileno,需使用fclose

#include <iostream>#include <cstdio>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>#define MY_FILE "log.txt"int main()
{umask(0);//close(stderr->_fileno);   这样会导致文件描述错误,资源泄漏等问题fclose(stderr); //正确关闭//或者直接//close(2);int fd1 = open(MY_FILE, O_RDWR | O_CREAT | O_TRUNC, 0666);printf("stdin fd -> %d\n", stdin->_fileno);printf("stdout fd -> %d\n", stdout->_fileno);printf("stderr fd -> %d\n", stderr->_fileno);printf("log.txt fd -> %d\n", fd1);return 0;
}

测试结果

         所以文件描述fd的分配规则是:从小到大,按照循环发方式找到文件描述符表中的最小且没有被占用的位置。

        假如我们关闭了1号文件fd,再去向stdout输出数据会发生什么事情?

测试代码:

#include <iostream>#include <cstdio>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>#define MY_FILE "log.txt"int main()
{umask(0);close(1);int fd1 = open(MY_FILE, O_RDWR | O_CREAT | O_TRUNC, 0666);int cnt = 10;while(cnt--){fprintf(stdout,"Hello YZC! [%d]\n",cnt);  //向stdout输出10条数据}return 0;
}

测试结果:

 

        可以看到,我们向屏幕输出数据转化为向log.txt这个文件中输出数据。 

        这种就是我们的文件输出重定向,由屏幕重定向到log.txt这个文件中

二. 文件重定向

2.1 ">"  ">>"  "<"  命令

        文件重定向广泛来说有三种,输出重定向,追加重定向,输入重定向。

        在Linux 中我们使用命令 > 即可完成输出重定向 >>完成追加重定向 < 完成输入重定向

 2.2 重定向的本质⭐

        我们知道,进程PCB通过文件描述符表找到并访问对应的文件

        重定向的本质就是C语言上层用的fd不改变,在内核中改变fd对应的truct_file*的地址。

        比如上面举例中,我们上层没有关闭stdout,而是关闭了stdout原本的标准输出fd(2号)。让后让指向log.txt的文件fd写入到stdout中。这样就完成了向log.txt的标准输出重定向。

2.3 使用dup2完成重定向 ⭐

        dup2可以复制文件描述符fd。函数原型如下

//所需头文件
#include <unistd.h>int dup2(int oldfd, int newfd);//解释
//将oldfd文件文件描述符fd拷贝到newfd
//或者说 将newfd内容更改为oldfd的内容//返回值
//失败返回-1,设置错误码。成功返回文件描述符

        注意:dup2不是简单的更改0,1,2... 。而是将指针数组中的0,1,2的内容struct_file更改

a dup2完成输出重定向

用法举例:

#include <iostream>#include <cstdio>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>#define MY_FILE "log.txt"int main()
{umask(0);int fd1 = open(MY_FILE, O_RDWR | O_CREAT | O_TRUNC, 0666);//将fd 1中的内容由指向标准输出更改为指向log.txtdup2(fd1,1);int cnt = 10;while(cnt--){fprintf(stdout,"Hello YZC! [%d]\n",cnt);  //向stdout输出10条数据}return 0;
}

 测试结果:

        可见,通过dup2我们完成了输出重定向

b dup2完成追加重定向

        想要完成追加重定向,只要将重定向文件的写入方式更改为追加即可

测试举例:

#include <iostream>#include <cstdio>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>#define MY_FILE "log.txt"int main()
{umask(0);//以追加的方式打开,而不是TRUNC清空int fd1 = open(MY_FILE, O_RDWR | O_CREAT | O_APPEND, 0666);//将fd 1中的内容由指向标准输出更改为指向log.txtdup2(fd1,1);int cnt = 10;while(cnt--){fprintf(stdout,"Hello YZC! [%d]\n",cnt);  //向stdout输出10条数据}return 0;
}

测试结果:

c dup2输入重定向 

        我们将文件描述符表中0的内容由标准输入转为由log.txt文件输入

#include <iostream>#include <cstdio>
#include <unistd.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>#define MY_FILE "log.txt"int main()
{// 以只读方式打开int fd1 = open(MY_FILE, O_RDONLY);// 更改0的中的内容,由标准输入变为由log.txt输入dup2(fd1, 0);char buffer[64];while (true){printf("请输入>");// 向buffer输入数据,若为空,直接跳出循环if (fgets(buffer, sizeof(buffer), stdin) == NULL)break;printf("%s", buffer);}return 0;
}

测试结果如下:

 三. 如何理解Linux下一切皆文件?

        我们知道,被打开的文件有文件描述结构体,这个结构体中包含了文件的各种属性。

struct file
{//文件的各种属性int type;int status;......//文件的读写方法指针int (*readp)();int (*write)();
}

        站在struct file的角度来说,各种文件或者设备(键盘,显示器,鼠标,磁盘等)统一都是 struct file。

        而我们用户通过struct file去调用其中的读写函数指针就能够调用具体的文件或者设备的读写方法。

        通过这种虚拟文件系统,我们就能够摒弃底层硬件的差别。而使用统一的视角去看待各种文件和硬件设备,使用统一的文件接口进行操作。所以说Linux下一切皆文件

        这种设计是不是也符合多态的原则

Linux中的源码 

四. 下篇内容:C语言FILE与用户级缓冲区与文件系统

http://www.lryc.cn/news/501800.html

相关文章:

  • k8s调度策略
  • uniapp中父组件传参到子组件页面渲染不生效问题处理实战记录
  • 螺丝螺帽缺陷检测识别数据集,支持yolo,coco,voc三种格式的标记,一共3081张图片
  • 一个简单带颜色的Map
  • kubeadm安装K8s集群之基础环境配置
  • 前端实现在线预览excel文件
  • 关于idea-Java-servlet-Tomcat-Web开发中出现404NOT FOUND问题的解决
  • SCRM私域流量管理工具助力企业微信电商转型升级
  • 三相异步电动机为什么能够旋转?
  • 优化移动端H5:常见问题与解决方案
  • TM1不藏私系列——#10. TM1快速运算的秘密武器-Feeder
  • 【Python】【Conda 】Conda vs venv:Python开发者的虚拟环境选择指南
  • 【从0学英语】06.时态 - 一般过去时
  • 获取cpu序列号-python实现
  • 文献分享: PLAID——为ColBERT架构设计的后期交互驱动器
  • IMX6ULL开发板、PC机上的USB网卡、VMware中的Ubuntu的桥接网卡三者互Ping设置及设置
  • 孚盟云 MailAjax.ashx SQL漏洞复现
  • 前端 mp4 视频改成 m3u8 流模式
  • 聚焦港口智能接处警,开启平安海运之门
  • `yarn list --pattern element-ui` 是一个 Yarn 命令,用于列出项目中符合指定模式(`element-ui`)的依赖包信息
  • ElementEye,网页分析器
  • 健康管理系统(Koa+Vue3)
  • 智创 AI 新视界 -- AI 助力金融风险管理的新策略(16 - 10)
  • Linux内核 -- 字符设备之read write poll基本实现
  • 腾讯微信C++面试题及参考答案
  • 如何查看内网设备访问互联网时的出口 IP 地址?
  • ESP32-S3模组上跑通ES8388(24)
  • 【AIGC系列】frequency_penalty如何通过控制参数提升文本生成的多样性与创造性
  • Python+OpenCV系列:图像的运算
  • 【Unity技巧】Unity项目中哪些文件不用管理(.gitignore)