Linux的基础IO:文件描述符 重定向本质

目录

前言

文件操作的系统调用接口

open函数

close函数

write函数

read函数 

注意事项

文件描述符-fd

小补充 

重定向

文件描述符的分配原则

系统调用接口-dup2

缓冲区

缓冲区的刷新策略

对于“2”的理解

小补充 


前言

        在Linux中一切皆文件,打开文件的本质是进程打开了文件,文件没有被打开时一直存放在磁盘中(进程执行时才会打开文件,文件才会从磁盘中拿出),而OS中存在很多进程,即系统中一定存在大量被进程打开的文件,对此OS会采取“先描述再组织”的原则,每个被打开的文件在OS内部都有一个类似PCB的描述文件属性的结构体。

文件 = 属性 + 内容

以w方式打开文件:

  • 文件不存在,则在当前路径下新建指定文件并写入
  • 文件存在,打开文件时会将该文件清空并写入

以a方式打开文件:

  • 文件不存在,则在当前路径下新建指定文件并写入
  • 文件存在,追加写入

>和>>:

  • > 文件 等同于 w 一个文件

  • >> 文件 等同于 a 一个文件

结论:输出重定向一定是文件操作

文件操作的系统调用接口

基本概念:文件在进程未执行时一直放在磁盘中,磁盘是硬件,向文件中写入本质就是向硬件中写入,但是用户没有权利直接向硬件中写入,需要由OS提供访问硬件的系统调用接口,而C/C++等编程语言中提供的对文件的操作接口就是对系统调用的接口的封装

open函数

函数原型: 

  • int open(const char *pathname, int flags);
  • int open(const char *pathname, int flags, mode_t mode);

包含头文件:

  • <sys/types.h>
  • <sys/stat.h>
  • <fcntl.h>

参数:

  • const char *pathname:要打开的文件绝对或相对路径
  • flags:位掩码,指定了打开方式和访问权限等信息
  • mode:指定新建文档的权限设置

flags的常见取值:

  • O_RDONLY:以只读模式打开
  • O_WRONLY:以只写模式打开
  • O_RDWR:以读写模式打开
  • O_CREAT:如若目标文档不存在则创建
  • O_TRUNC:如若目标文档存在则清空
  • O_APPEND:追加写入

返回值:操作文件成功时返回int类型的文件描述符,失败时返回 -1

功能:打开文件的系统调用函数

close函数

函数原型:int close(int fd);

包含头文件: <unistd.h>

参数:要关闭的目标文件描述符或套接字

返回值:关闭成功时返回 0,失败时返回 -1 

功能:关闭指定文件描述符或套接字,并释放与之相关联的资源

write函数

函数原型:ssize_t write(int fd, const void *buf, size_t count);

包含头文件: <unistd.h>

参数:待写入的目标文件描述符或套接字,待写入数据的缓冲区指针,待写入的字节数

返回值:写文件成功时返回实际写入到目标文件中的字节数,失败时返回 -1

功能:将buf指向的内容写入到目标文件描述符或套接字所对应对象中

read函数 

函数原型:ssize_t write(int fd, const void *buf, size_t count);

包含头文件: <unistd.h>

参数:待写入的目标文件描述符或套接字,待写入数据的缓冲区指针,待写入的字节数

返回值:写文件成功时返回实际写入到目标文件中的字节数,失败时返回 -1

功能:将buf指向的内容写入到目标文件描述符或套接字所对应对象中

        stat、fastat、lasta是三个修改文件属性的调用接口,上面的read、write等是对文件内容修改的调用接口

注意事项

1、文件的权限掩码的采用就近原则,默认为0002,uamsk(0)设置为0后就为0

2、O_RDONLY等位掩码都是类似于下列形式的宏定义(了解)

3、位掩码不同的组合有不同的效果

  • O_WRONLY | O_CREAT | O_TRUNC:实现fopen函数w打开文件时的效果
  • O_WRONLY | O_CREAT | O_APPEND:实现fopen函数a打开文件时的效果
  • O_WRONLY | O_CREAT:实现fopen函数wa打开文件时的效果

文件描述符-fd

问题一:open函数的返回值是文件描述符,我们创建四个文件并打印它们的文件描述符发现它们分别是3、4、5、6,怎么没有见0、1、2?

原因:C语言在运行时会默认打开三个流,而0、1、2就分别是标准输入流stdin(键盘)、标准输出流stdout(显示器)、标准错误流stderr(显示器)

问题二:为什么可以向1中写?fd的本质是什么 

结论:文件描述符的本质是内核的进程的文件映射关系数组的下标

小补充 

       程序的本质确实是对数据进行处理,并且这些处理过程和结果需要与人类用户进行交互。在Unix/Linux系统中,0、1、2三个文件描述符分别代表标准输入、标准输出和标准错误输出。它们被默认打开并与终端设备关联,以便程序可以通过它们与用户进行交互

问题三:read和write函数如何进行读写文件?

解释:read时将文件存放在内核级缓存中的数据拷贝至上层,如果内核级缓存中没有文件的数据,就将要read的进程挂在磁盘的等待队列中,等待磁盘将相应的文件的数据放入内核级缓存后再唤醒该进程并进行拷贝,write时也是针对文件内核级缓冲区中的数据进行修改,修改后再刷新至磁盘中,因而无论读写都要在合适的时候,让OS将文件的数据读取到文件内核级缓冲区中,读写的对象都是文件内核级缓冲区中的数据,而不是磁盘级的文件数据

问题四:open函数在干什么?

解释:

  1. 创建文件
  2. 开辟内核级文件缓冲区的空间,加载文件数据(有延后性)
  3. 查看进程的文件描述符表(struct file_struct)
  4. 获取文件地址,并填入文件描述符表中
  5. 返回该文件在文件描述符表中的下标

问题五:为什么0、1、2在程序启动时默认打开?它们对应不应该是硬件吗?

补充: 硬件设备也有的struct_file,但是它们的struct_file中除了有硬件设备相关的属性外,还都包含一张指向底层操作方法(对硬件设备的操作接口)的函数指针表(每张表都一样),该表中的函数指针指向的是由硬件生产厂商的开发人员在驱动层已经写好的硬件调用接口,并且一个函数指针可以指向多个不同的硬件操作接口从而产生不同的结果(多态,write函数指针可以指向键盘、鼠标等硬件的write接口,这一功能由厂商实现的)

解释:因为在进程执行时就会将三个硬件的struct_file的地址信息放入进程struct files_struct中,数组下标0、1、2就是它们的文件标识符,而它们的struct_file中又有指向底层操作方法的函数指针,在进程尝试使用硬件设备时会通过它们的文件标识符访问到它们的struct_file进而访问到驱动层的k_read()等硬件设备操作接口

注意事项:普通文件类型的 struct file 中并不包含指向底层设备接口的函数指针表

        struct_file中的文件属性就是类中的数据,操作底层方法的指针表中的函数指针就是类中的方法,所以struct_file也可以视为C语言实现的类,多个struct_file构成了OS中的virtual file system

问题六:如何理解C语言通过FILE* 访问文件?

解释:FILE是一个由文件描述符等内容封装成的结构体,C语言中所有文件操作函数,都是对系统调用接口的封装,fopen函数是对调用接口open的封装,该函数隐式返回了文件描述符给FILE结构体,而fwrite、fread等又是对write、read调用接口的封装,fwrite等可以通过FILE中的fd访问文件

好文章:文件操作的底层原理(文件描述符与缓冲区) - 知乎 (zhihu.com)  

重定向

文件描述符的分配原则

基本概念:查自己的文件描述表,分配最小的没有被使用的fd

尝试为一个普通文件分配fd = 1,并调用printf和fprintf函数向显示器上打印内容:

现象: 本来应该打印到显示器上的内容,却打印到了一个指定的文件中,这种技术叫重定向

结论:重定向的本质就是在内核中改变文件描述符表特定下标的内容,与上层无关,重定向也可以视为对open和dup2接口的封装:

int fd = open("output.txt", O_WRONLY | O_CREAT | O_TRUNC, 0644);
dup2(fd, STDOUT_FILENO); //原 > 目标

如果想要实现追加重定向>>,那么就只需更改open函数的参数:

int fd = open("output.txt", O_WRONLY | O_CREAT | O_APPEND, 0644);
dup2(fd, STDOUT_FILENO); //原 > 目标

接着尝试注释fflush,注释fflush和close、注释close:

解释:stdin、stdout、stderr的struct_file中除了有_fileno还有语言级别的文件缓冲区,printf和fprintf函数会先将要打印的内容放入stdoutd的struct_file中的文件缓冲区中,由fflush(stdout)指令将该缓冲区中的数据刷新至log.txt的内核文件缓冲区中(因为此时fd = 1指向的是log.txt),最后由OS定期将log.txt内核级文件缓冲区中的数据刷新至磁盘中:注释fflush就会导致stdout的struct_file中的文件缓冲区数据无法刷新至内核级的文件缓冲区并且文件描述符还被关闭了即使在最后操作系统想要帮助刷新也不可能了(如果你没有显式地刷新 stdout 缓存或关闭该进程所占据的资源,则操作系统可能会自动地执行这些操作以确保程序正常结束并释放相关资源,跟\n没关系,即使你这里将\n去除也不会刷新缓冲区只有stdout与显示器关联时才能刷新,\n在这里只有换行作用,本质是行刷新和全缓冲刷新的不同,刷新策略中有解释)

结论:fflush(stdout)是为了将stdout的语言级文件缓冲区中的内容刷新至内核级文件缓冲区中

系统调用接口-dup2

函数原型:int dup2(int oldfd,int newfd);

包含头文件:<unistd.h>

参数:源文件描述符,目标文件描述符

返回值:成功返回目标文件描述符,失败返回-1

功能:使得目标文件描述符共享源文件描述符所对应的 数据

缓冲区

基本概念:缓冲区分为用户级(语言级)缓冲区和内核级缓冲区,是一段内存空间

优点:缓冲区的分级有利于解耦、提高使用者的效率、提高刷新IO效率

问题:为什么C语言可以通过调用接口直接向内核级缓冲区写数据,还要提供对调用接口重新封装后的接口并先将数据写到语言级缓冲区呢?

解释:调用系统接口是有成本的,多次频繁的使用write等系统调用接口向内核级缓冲区中写一些很少的数据会造成资源浪费,而使用了封装后的fwrite接口就可以先将这些内容放入语言级缓冲区中,放入之后就可以结束fwrite进行下一步操作等语言级的缓冲区中的数据达到一定程度时仅调用一次系统调用接口,就可以将多次写入的少量数据一次性的放入内核级缓冲区中并且写入后向磁盘刷新的操作也有OS自行完成不需要用户管,这样就即提高了使用者的效率(把快递交给快递员而不是亲自去送,交给后就不用管了可以去干其它内容),又提高了刷新IO的效率(将一段时间内的所有接收到的快递装车一块去配送,而不是接受一件配送一件,并且应该由专门的快递配送员配送,相比于寄件人他们知道配送的流程,寄件人了乐于有人帮自己干活)

缓冲区的刷新策略

对于用户 / 内核级的缓冲区都适用,但是这里我们只关心用户级的

立即刷新(近似于无缓冲)

1、用户级接口:fflush(stdout)强制刷新用户级的缓冲区至内核级缓冲区

2、内核级接口:int fsync(int fd)强制刷新内核级的缓冲区至磁盘

行刷新

显示器的行刷新是为了便于用户观看数据

全缓冲

缓冲区写满才刷新,一般是普通文件(此时\n只起到换行作用)

特殊情况

1、进程退出,系统会自动刷新

2、强制刷新

注意事项:

1、不同平台的刷新策略不同

2、子进程不会继承父进程在用户级缓冲区中刷新过的内容

关于完善shell中重定向的内容在22的2:30处,一小时左右

对于“2”的理解

基本概念:1和2中的内容都是显示器文件的struct file

问题一:为什么要有2?

解释:分离程序中出现的正确和错误的消息,正确的信息向1中打,错误的信息向2中打,再通过重定向建立存放正确和错误信息的两个文件

问题二:>是标准输出重定向,只会更改1号fd中的内容,如何将2和1定向至同一文件中?

解释:. / a.out 1>all.log 2>&1,先将1获取到的正确内容放入all.log文件中,然后将2获取到的错误的内容放入1放入的文件中(由取地址&实现)

小补充 

perror函数本质上是向2中打印,printf本质上是向1中打印的:

~over~

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/831217.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Leetcode 108.将有序数组转换为二叉搜索树

题目描述 给你一个整数数组 nums &#xff0c;其中元素已经按 升序 排列&#xff0c;请你将其转换为一棵 平衡 二叉搜索树。 示例 1&#xff1a; 输入&#xff1a;nums [-10,-3,0,5,9] 输出&#xff1a;[0,-3,9,-10,null,5] 解释&#xff1a;[0,-10,5,null,-3,null,9] 也将被…

改变 centos yum源 repo

centos 使用自带的 repo 源 速度慢&#xff0c;可以改为国内的&#xff0c;需要改两个地方 centos7.repo CentOS-Base.repo 首先备份/etc/yum.repos.d/CentOS-Base.repo mv /etc/yum.repos.d/CentOS-Base.repo /etc/yum.repos.d/CentOS-Base.repo.backup下载对应版本repo文件…

ICMP详解

3 ICMP ICMP&#xff08;Internet Control Message Protocol&#xff0c;因特网控制报文协议&#xff09;是一个差错报告机制&#xff0c;是TCP/IP协议簇中的一个重要子协议&#xff0c;通常被IP层或更高层协议&#xff08;TCP或UDP&#xff09;使用&#xff0c;属于网络层协议…

Uniapp好看登录注册页面

个人介绍 hello hello~ &#xff0c;这里是 code袁~&#x1f496;&#x1f496; &#xff0c;欢迎大家点赞&#x1f973;&#x1f973;关注&#x1f4a5;&#x1f4a5;收藏&#x1f339;&#x1f339;&#x1f339; &#x1f981;作者简介&#xff1a;一名喜欢分享和记录学习的…

4G+北斗太阳能定位终端:一键报警+倾覆报警 双重保障船舶安全

海上作业环境复杂多变&#xff0c;海上航行充满了各种不确定性和风险&#xff0c;安全事故时有发生&#xff0c;因此海上安全与应急响应一直是渔业和海运行业关注的重点。为了提高海上安全保障水平&#xff0c;4G北斗太阳能定位终端应运而生&#xff0c;它集成了一键报警和倾覆…

Edge浏览器新特性深度解析,写作ai免费软件

首先&#xff0c;这篇文章是基于笔尖AI写作进行文章创作的&#xff0c;喜欢的宝子&#xff0c;也可以去体验下&#xff0c;解放双手&#xff0c;上班直接摸鱼~ 按照惯例&#xff0c;先介绍下这款笔尖AI写作&#xff0c;宝子也可以直接下滑跳过看正文~ 笔尖Ai写作&#xff1a;…

【数据结构】:链表的带环问题

&#x1f381;个人主页&#xff1a;我们的五年 &#x1f50d;系列专栏&#xff1a;数据结构 &#x1f337;追光的人&#xff0c;终会万丈光芒 前言&#xff1a; 链表的带环问题在链表中是一类比较难的问题&#xff0c;它对我们的思维有一个比较高的要求&#xff0c;但是这一类…

AI大模型探索之路-训练篇10:大语言模型Transformer库-Tokenizer组件实践

系列篇章&#x1f4a5; AI大模型探索之路-训练篇1&#xff1a;大语言模型微调基础认知 AI大模型探索之路-训练篇2&#xff1a;大语言模型预训练基础认知 AI大模型探索之路-训练篇3&#xff1a;大语言模型全景解读 AI大模型探索之路-训练篇4&#xff1a;大语言模型训练数据集概…

DS:顺序表、单链表的相关OJ题训练

欢迎各位来到 Harper.Lee 的学习小世界&#xff01; 博主主页传送门&#xff1a;Harper.Lee的博客主页 想要一起进步的uu可以来后台找我交流哦&#xff01; 在DS&#xff1a;单链表的实现 和 DS&#xff1a;顺序表的实现这两篇文章中&#xff0c;我详细介绍了顺序表和单链表的…

使用LinkAI创建AI智能体,并快速接入到微信/企微/公众号/钉钉/飞书

​ LinkAI 作为企业级一站式AI Agent 智能体搭建与接入平台&#xff0c;不仅为用户和客户提供能够快速搭建具备行业知识和个性化设定的 AI 智能体的能力&#xff1b;还基于企业级场景提供丰富的应用接入能力&#xff0c;让智能体不再是“玩具”&#xff0c;而是真正能够落地应用…

PHP的数组练习实验

实 验 目 的 掌握索引和关联数组&#xff0c;以及下标和元素概念&#xff1b; 掌握数组创建、初始化&#xff0c;以及元素添加、删除、修改操作&#xff1b; 掌握foreach作用、语法、执行过程和使用&#xff1b; 能应用数组输出表格和数据。 任务1&#xff1a;使用一维索引数…

uniapp0基础编写安卓原生插件和调用第三方jar包和编写语音播报插件之使用jar包插件

前言 如果你不会编写安卓插件,你可以先看看我之前零基础的文章(uniapp0基础编写安卓原生插件和调用第三方jar包和编写语音播报插件之零基础编写安卓插件), 我们使用第三方包,jar包编写安卓插件 开始 把依赖包,放到某个模块的/libs目录(myTestPlug/libs) 还要到build…

R语言的学习—5—多元数据直观表示

1、数据读取 ## 数据整理 d3.1read.xlsx(adstats.xlsx,d3.1,rowNamesT);d3.1 #读取adstats.xlsx表格d3.1数据 barplot(apply(d3.1,1,mean)) #按行做均值条形图 barplot(apply(d3.1,1,mean),las3) barplot(apply(d3.1,2,mean)) #按列做均值图条形图 barplot(a…

C语言数据结构 ---- 单链表实现通讯录

今日备忘录: "折磨我们的往往是想象, 而不是现实." 目录 1. 前言2. 通讯录的功能3. 通讯录的实现思路5. 效果展示6. 完整代码7. 总结 正文开始 1. 前言 顺表实现通讯录: 点击~ 顺序表实现通讯录 在日常生活中&#xff0c;我们经常需要记录和管理大量的联系人信息&…

【研发管理】产品经理知识体系-组合管理

导读&#xff1a;新产品开发的组合管理是一个重要的过程&#xff0c;它涉及到对一系列新产品开发项目进行策略性选择、优先级排序、资源分配和监控。这个过程旨在确保企业能够最大化地利用有限的资源&#xff0c;以实现其战略目标。 目录 1、组合管理、五大目标 2、组合管理的…

第74天:漏洞发现-Web框架中间件插件BurpSuite浏览器被动主动探针

目录 思维导图 前置知识 案例一&#xff1a;浏览器插件-辅助&资产&漏洞库-Hack-Tools&Fofa_view&Pentestkit 案例二&#xff1a; BurpSuite 插件-被动&特定扫描-Fiora&Fastjson&Shiro&Log4j 思维导图 前置知识 目标&#xff1a; 1. 用…

基于springboot实现公司日常考勤系统项目【项目源码+论文说明】

基于springboot实现公司日常考勤系统演示 摘要 目前社会当中主要特征就是对于信息的传播比较快和信息内容的安全问题&#xff0c;原本进行办公的类型都耗费了很多的资源、传播的速度也是相对较慢、准确性不高等许多的不足。这个系统就是运用计算机软件来完成对于企业当中出勤率…

数据结构-链表OJ

1.删除链表中等于给定值 val 的所有结点。 . - 力扣&#xff08;LeetCode&#xff09; 思路一&#xff1a;遍历原链表&#xff0c;将值为val的节点释放掉 思路二&#xff1a;创建一个新链表&#xff0c;将值不为val的节点尾插到新链表中 /*** Definition for singly-linked …

2024年五一数学建模竞赛C题论文首发

基于随机森林的煤矿深部开采冲击地压危险预测 摘要 煤炭作为中国重要的能源和工业原料&#xff0c;其开采活动对国家经济的稳定与发展起着至关重要的作用。本文将使用题目给出的数据探索更为高效的数据分析方法和更先进的监测设备&#xff0c;以提高预警系统的准确性和可靠性…

智能消费记账|基于SSM+vue的大学生智能消费记账系统(源码+数据库+文档)

智能消费记账目录 基于SSMvue的大学生智能消费记账系统 一、前言 二、系统设计 三、系统功能设计 1 用户列表 2 预算信息管理 3 预算类型管理 四、数据库设计 五、核心代码 六、论文参考 七、最新计算机毕设选题推荐 八、源码获取&#xff1a; 博主介绍&#xff1…