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与用户级缓冲区与文件系统
相关文章:
Linux操作系统3-文件与IO操作2(文件描述符fd与文件重定向)
上篇文章:Linux操作系统3-文件与IO操作1(从C语言IO操作到系统调用)-CSDN博客 本篇代码Gitee仓库:myLerningCode 橘子真甜/Linux操作系统与网络编程学习 - 码云 - 开源中国 (gitee.com) 本篇重点:文件描述符fd与文件重定向 目录 一. 文件描述…...
k8s调度策略
调度策略 binpack(装箱策略) Binpacking策略(又称装箱问题)是一种优化算法,用于将物品有效地放入容器(或“箱子”)中,使得所使用的容器数量最少,Kubernetes等集群管理系…...
uniapp中父组件传参到子组件页面渲染不生效问题处理实战记录
上篇文件介绍了,父组件数据更新正常但是页面渲染不生效的问题,详情可以看下:uniapp中父组件数组更新后与页面渲染数组不一致实战记录 本文在此基础上由于新增需求衍生出新的问题.本文只记录一下解决思路. 下面说下新增需求方便理解场景: 商品信息设置中添加抽奖概率设置…...
螺丝螺帽缺陷检测识别数据集,支持yolo,coco,voc三种格式的标记,一共3081张图片
螺丝螺帽缺陷检测识别数据集,支持yolo,coco,voc三种格式的标记,一共3081张图片 3081总图像数 数据集分割 训练组90% 2781图片 有效集7% 220图片 测试集3% 80图片 预处理…...
一个简单带颜色的Map
越简单 越实用。越少设计,越易懂。 需求背景: 创建方法,声明一个hashset, 元素为 {“#DE3200”, “#FA8C00”, “#027B00”, “#27B600”, “#5EB600”} 。 对应的key为 key1 、key2、key3、key4、key5。 封装该方法,…...
kubeadm安装K8s集群之基础环境配置
系列文章目录 1.kubeadm安装K8s集群之基础环境配置 2.kubeadm安装K8s集群之高可用组件keepalivednginx 3.kubeadm安装K8s集群之master节点加入 4.kubeadm安装K8s集群之worker1节点加入 kubeadm安装K8s集群基础环境配置 1.首先确保所有机器可以通信,然后配置主机host…...
前端实现在线预览excel文件
在前端开发中,经常会遇到需要在线预览各种文件的需求。本文将介绍如何使用前端技术实现在线预览 Excel 文件的功能。 一、基于微软office服务的excel预览 获取要预览的 Excel 文件的 URL(例如存储在 OneDrive 或 SharePoint 上的文件)。 使…...
关于idea-Java-servlet-Tomcat-Web开发中出现404NOT FOUND问题的解决
在做web项目时,第一次使用servlet开发链接前端和后端的操作,果不其然,遇到了诸多问题,而遇到最多的就是运行项目打开页面时出现404NOT FOUND的情况。因为这个问题我也是鼓捣了好久,上网查了许多资料才最终解决…...
SCRM私域流量管理工具助力企业微信电商转型升级
内容概要 在当今数字化时代,SCRM(社交客户关系管理)私域流量管理工具正逐渐成为企业转型的重要助力。尤其是在电商领域,企业微信的兴起为许多公司打开了新的销售渠道,通过SCRM系统的高效整合,企业能够更加…...
三相异步电动机为什么能够旋转?
三相异步电动机,作为一种广泛应用于工业、农业及其他领域的电动机,其工作原理的理解对于工程技术人员以及相关从业者来说至关重要。 一、三相异步电动机的基本结构 三相异步电动机主要由定子、转子和机壳组成。定子是电动机的静止部分,包含…...
优化移动端H5:常见问题与解决方案
移动端H5开发中的“坑”与解决方案 本文介绍了开发中遇到的几个关于移动端H5开发中的小问题,以及解决的方法。 一、iOS滑动不流畅问题 在iOS设备上,H5页面的滑动效果有时会出现不流畅的情况,特别是在页面高度超过一屏时。这通常是由于iOS的…...
TM1不藏私系列——#10. TM1快速运算的秘密武器-Feeder
与其他BI产品对比,TM1的快速运算能力一骑绝尘。 但是在多维度的数据组合下,TM1是依据什么进行运算的呢? 今天将和大家一同了解TM1快速运算的秘密武器-Feeder。 上期我们提到通过配置维度中的元素权重,可以在合并层级加总计算。除…...
【Python】【Conda 】Conda vs venv:Python开发者的虚拟环境选择指南
目录 引言一、概述1.1 Conda 虚拟环境1.2 Python venv 虚拟环境 二、安装与设置2.1 安装 Conda 虚拟环境2.2 安装 Python venv 虚拟环境 三、依赖管理3.1 Conda 依赖管理3.2 Python venv 依赖管理 四、适用场景五、性能与资源占用5.1 Conda 性能与资源占用5.2 Python venv 性能…...
【从0学英语】06.时态 - 一般过去时
一般过去时(Past Simple Tense)是表达过去发生的动作、状态或事实的核心时态。这一时态都扮演着不可或缺的角色,本篇文章将全面讲解一般过去时的定义、结构、用法以及常见的动词变化,通过例句和详细的解释帮你理解这一时态。 文章…...
获取cpu序列号-python实现
DataBall 助力快速掌握数据集的信息和使用方式,会员享有 百种数据集,持续增加中。 需要更多数据资源和技术解决方案,知识星球: “DataBall - X 数据球(free)” -------------------------------------------------------------…...
文献分享: PLAID——为ColBERT架构设计的后期交互驱动器
👉前情提要: 神经网络自然语言模型概述 Transformer \text{Transformer} Transformer与注意力机制概述 📚相关论文: BERT: Pre-training of Deep Bidirectional Transformers for Language Understanding \text{BERT: Pre-train…...
IMX6ULL开发板、PC机上的USB网卡、VMware中的Ubuntu的桥接网卡三者互Ping设置及设置
连上PC机上的USB网卡配置 首先打开Windows设备管理器,截图记录下当前的网络适配器,作为插上USB网卡后的对比: 然后打开“更改适配器选项”,也截张图,作为插上USB网卡后的对比: 插上USB网口࿰…...
孚盟云 MailAjax.ashx SQL漏洞复现
0x01 产品描述: 孚盟云是由...
前端 mp4 视频改成 m3u8 流模式
前端 mp4 视频改成 m3u8 流模式 mp4 视频的问题 1、mp4 视频通常对应一个文件,播放时需要加载全部文件,消耗网络资源。如果用户从中间某个时间访问,也会从头开始下载,浪费服务器性能。 2、mp4 视频文件容易被用户下载到本地。有…...
聚焦港口智能接处警,开启平安海运之门
一、概述 港口作为海运的关键枢纽,安全保障至关重要。港口智能接处警系统凭借智能化、数字化、信息化手段,融合 AI 与大数据等前沿技术,旨在实现对港口各类警情的快速、精准处理,有效提升港口应急响应能力,开启平安海…...
在 Nginx Stream 层“改写”MQTT ngx_stream_mqtt_filter_module
1、为什么要修改 CONNECT 报文? 多租户隔离:自动为接入设备追加租户前缀,后端按 ClientID 拆分队列。零代码鉴权:将入站用户名替换为 OAuth Access-Token,后端 Broker 统一校验。灰度发布:根据 IP/地理位写…...
[ICLR 2022]How Much Can CLIP Benefit Vision-and-Language Tasks?
论文网址:pdf 英文是纯手打的!论文原文的summarizing and paraphrasing。可能会出现难以避免的拼写错误和语法错误,若有发现欢迎评论指正!文章偏向于笔记,谨慎食用 目录 1. 心得 2. 论文逐段精读 2.1. Abstract 2…...
CMake 从 GitHub 下载第三方库并使用
有时我们希望直接使用 GitHub 上的开源库,而不想手动下载、编译和安装。 可以利用 CMake 提供的 FetchContent 模块来实现自动下载、构建和链接第三方库。 FetchContent 命令官方文档✅ 示例代码 我们将以 fmt 这个流行的格式化库为例,演示如何: 使用 FetchContent 从 GitH…...
【Android】Android 开发 ADB 常用指令
查看当前连接的设备 adb devices 连接设备 adb connect 设备IP 断开已连接的设备 adb disconnect 设备IP 安装应用 adb install 安装包的路径 卸载应用 adb uninstall 应用包名 查看已安装的应用包名 adb shell pm list packages 查看已安装的第三方应用包名 adb shell pm list…...
FFmpeg avformat_open_input函数分析
函数内部的总体流程如下: avformat_open_input 精简后的代码如下: int avformat_open_input(AVFormatContext **ps, const char *filename,ff_const59 AVInputFormat *fmt, AVDictionary **options) {AVFormatContext *s *ps;int i, ret 0;AVDictio…...
Kubernetes 节点自动伸缩(Cluster Autoscaler)原理与实践
在 Kubernetes 集群中,如何在保障应用高可用的同时有效地管理资源,一直是运维人员和开发者关注的重点。随着微服务架构的普及,集群内各个服务的负载波动日趋明显,传统的手动扩缩容方式已无法满足实时性和弹性需求。 Cluster Auto…...
AxureRP-Pro-Beta-Setup_114413.exe (6.0.0.2887)
Name:3ddown Serial:FiCGEezgdGoYILo8U/2MFyCWj0jZoJc/sziRRj2/ENvtEq7w1RH97k5MWctqVHA 注册用户名:Axure 序列号:8t3Yk/zu4cX601/seX6wBZgYRVj/lkC2PICCdO4sFKCCLx8mcCnccoylVb40lP...
02.运算符
目录 什么是运算符 算术运算符 1.基本四则运算符 2.增量运算符 3.自增/自减运算符 关系运算符 逻辑运算符 &&:逻辑与 ||:逻辑或 !:逻辑非 短路求值 位运算符 按位与&: 按位或 | 按位取反~ …...
用递归算法解锁「子集」问题 —— LeetCode 78题解析
文章目录 一、题目介绍二、递归思路详解:从决策树开始理解三、解法一:二叉决策树 DFS四、解法二:组合式回溯写法(推荐)五、解法对比 递归算法是编程中一种非常强大且常见的思想,它能够优雅地解决很多复杂的…...
ThreadLocal 源码
ThreadLocal 源码 此类提供线程局部变量。这些变量不同于它们的普通对应物,因为每个访问一个线程局部变量的线程(通过其 get 或 set 方法)都有自己独立初始化的变量副本。ThreadLocal 实例通常是类中的私有静态字段,这些类希望将…...
