【Linux文件管理】重定向内核级缓冲区用户级缓冲区

在这里插入图片描述

文章目录

  • 文件管理
    • 文件描述符表(files_struct)
    • 重定向
      • 重定向概念
  • 文件的内核级缓冲区和用户级缓冲区
    • 内核级缓冲区
    • 用户级缓冲区
  • 总结

文件管理

文件描述符表(files_struct)

上一期我们将文件描述符讲完了,这期来讲讲文件管理中的文件描述符表,在task_struct有一个指针是指向文件描述符表的。
在这里插入图片描述
我们进入进入文件描述符表:
在这里插入图片描述
可以看到当中有很多属性,虽然这些我们都不知道,但是有一个我么是知道的,就是红框框起来的,这个我们是知道的,这个是一个文件的指针数组,这个数组的下标就是文件描述符。
我们来画一个形象的结构。
在这里插入图片描述
文件描述符表的结构如图所示,我们将其余属性给屏蔽了,只留下一个指向文件的指针数组,这个file的指针数组,内存存储的是文件的属性,不仅是大小和路径还有很多属性。
有了文件描述符表,进程和文件就可以通过文件描述符表来管理了。
我们先简单写一段代码:

#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<unistd.h>
#include<fcntl.h>
int main()
{//打开文件int fd = open("log.txt",O_WRONLY | O_CREAT | O_TRUNC,0666);printf("fd:fileno:%d\n",fd);close(fd);return 0;
}

在这里插入图片描述
这段代码输出的是3,意思就是在数组中的位置就是3,为什么呢?

原因:因为当我们运行C语言程序的时候会自动启动三个流,分别是标准输出流,标准输入流,标准错误流。这三个流分别占了数组的0,1,2这三个位置,我们可以将这三个流的文件描述符打印一下。
在这里插入图片描述

通过这个示例我们可以画出下面的图:
在这里插入图片描述
前三个是运行程序的时候默认打开的三个流,假如我们任意关闭一个会出现什么状况呢,我们打开的新的文件会去占据腾出来的那个空位吗?
代码:

#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<unistd.h>
#include<fcntl.h>
int main()
{close(0);//打开文件int fd = open("log.txt",O_WRONLY | O_CREAT | O_TRUNC,0666);printf("fd:fileno:%d\n",fd);printf("stdin:fileno:%d\n",stdin->_fileno);printf("stdout:fileno:%d\n",stdout->_fileno);printf("stderr:fileno:%d\n",stderr->_fileno);close(fd);return 0;
}

我们提前将0对应的输入流给关掉了,可以可以看见结果,我们打开的新的文件去占据了以前的0的位置。
在这里插入图片描述
我们试试关闭输出流:
在这里插入图片描述
当我们关闭输出流的时候屏幕上是不会打印的,因为我们将输出流给关闭了,所以不会在屏幕上打印,又因为我们打开的文件占据了以前输出流数组下标对应的位置,所以不会打印在屏幕上,会打印在文件中,我们来查看一下:
在这里插入图片描述
可以看见文件中也没有,这里先不解释,这里其实存在一个用户级缓冲区,还没刷新到内核缓冲区当中,就将文件关闭了,所以这里不会写入到文件当中。
这里只需要强制刷新一下即可:

#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<unistd.h>
#include<fcntl.h>
int main()
{close(1);//打开文件int fd = open("log.txt",O_WRONLY | O_CREAT | O_TRUNC,0666);printf("fd:fileno:%d\n",fd);printf("stdin:fileno:%d\n",stdin->_fileno);printf("stdout:fileno:%d\n",stdout->_fileno);printf("stderr:fileno:%d\n",stderr->_fileno);fflush(stdout);close(fd);return 0;
}

在这里插入图片描述
这里可以看到刷新了一下之后就会打印在文件当中。
这里就引入一个概念:重定向

重定向

重定向概念

概念:操作系统中的一种机制,用于将程序的输入或输出流从默认位置(通常是终端)改变到其他位置(如文件或设备)。它通过操作文件描述符来实现,在 Linux 和 Unix 系统中非常常见。
用一张图来表示一下文件描述符:
在这里插入图片描述
简单来说这就是重定向,原本1是指向标准输出流的,但是将指向改变,将1指向新的文件,这就是重定向。

重定向函数:dup2
在这里插入图片描述

这是一个重定向函数,我们只看dup2,
在这里插入图片描述
在这里插入图片描述
dup2这个函数的作用就是将newfd关闭,然后将oldfd指向的file用newfd指向。
我们写一段简单的代码:

#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<unistd.h>
#include<fcntl.h>
int main()
{int fd = open("log.txt",O_WRONLY | O_CREAT | O_TRUNC,0666);int newfd = dup2(fd,1);//重定向if(newfd == -1) perror("dup2");printf("hello dup\n");return 0;
}

可以看见没有打印到屏幕上,而是写入到了文件中。
在这里插入图片描述

文件的内核级缓冲区和用户级缓冲区

内核级缓冲区

我们所用的write和read都不是直接写入到文件当中或者直接写入到外设当中。
在这中间还有一个内核级缓冲区,需要经过内核级缓冲区,最后写入到外设由外设自主决定,这里画一个简图来描述一下write的过程:
在这里插入图片描述
当我们调用write的时候,其实是将字符串拷贝到内核级缓冲区,然后由内核自主决定是否刷新到外设当中。

read的过程:
在这里插入图片描述
调用read的时候其实也是先从磁盘中读入到内核级缓冲区当中,最后拷贝到文件当中。
下面是Linux原码:
在这里插入图片描述

用户级缓冲区

用户级缓冲区:应用程序在用户空间中为存储和操作数据而分配的内存区域。与内核级缓冲区不同,用户级缓冲区完全由用户程序控制,内核不会直接干预这些缓冲区的管理。用户级缓冲区通常用于提高应用程序的性能,避免每次 I/O 操作时都直接与操作系统或外设进行交互,从而降低性能损耗。
回到之前的问题:

#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<unistd.h>
#include<fcntl.h>
int main()
{close(1);//打开文件int fd = open("log.txt",O_WRONLY | O_CREAT | O_TRUNC,0666);printf("fd:fileno:%d\n",fd);printf("stdin:fileno:%d\n",stdin->_fileno);printf("stdout:fileno:%d\n",stdout->_fileno);printf("stderr:fileno:%d\n",stderr->_fileno);fflush(stdout);close(fd);return 0;
}

会什么这里没有fflush就不会刷新到文件当中呢?
这里如果没有close是会打印的,或者将close换乘fclose也会刷新到文件当中。原因是因为:如果我们直接close,printf所打印的内讧还在用户级缓冲区当中,还没有刷新到文件当中,原本应该打印到屏幕上,但是1关闭了,所以重定向到文件当中,但是屏幕的刷新方式是按行刷新,也就是按照\n刷新,但是这里重定向之后就是按照文件的刷新方式来进行,也就是全刷新,等文件满了才刷新,但是这里没满所以不会刷新,所以直接close时,还没有刷新到文件当中,但是如果不close,程序退出前会自动刷新。
为什么会存在用户级缓冲区呢?

用户级缓冲区的存在是为了优化程序的 I/O 性能,减少频繁的系统调用开销,同时提高系统的响应速度和效率。

在这里插入图片描述
从用户级缓冲区强制刷新到内核级缓冲区叫fflush,从内核级缓冲区强制刷新到外设当中叫fsync
fsync:
在这里插入图片描述

总结

通过本文对文件管理及其内核级与用户级缓冲区的详细探讨,我们对操作系统中文件管理的机制有了更深刻的理解。首先,我们了解了 files_struct 的作用及其如何在内核中管理文件描述符的详细实现,掌握了文件的重定向以及文件描述符的相关操作。其次,通过对内核级缓冲区与用户级缓冲区的对比分析,我们认识到这两者在性能优化和内存管理中的关键作用,特别是在提升 I/O 性能和减少系统调用开销方面的重要性。

文件管理作为操作系统中非常核心的部分,不仅直接影响着系统资源的利用效率,也对程序的执行性能和稳定性起着至关重要的作用。通过合理管理内核级缓冲区和用户级缓冲区,操作系统能够有效地平衡性能和资源消耗,在确保数据准确性的同时提升系统的响应速度和吞吐量。

总之,深入理解操作系统中的文件管理机制,能够帮助我们更好地设计和优化应用程序,提升系统的整体效率。未来,随着操作系统的不断发展和优化,文件管理机制将会面临更多新的挑战和机遇,我们也将继续探索和学习,以应对新的技术发展。

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

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

相关文章

【H2O2|全栈】Node.js与MySQL连接

目录 前言 开篇语 准备工作 初始配置 创建连接池 操作数据库 封装方法 结束语 前言 开篇语 本节讲解如何使用Node.js实现与MySQL数据库的连接&#xff0c;并将该过程进行函数封装。 与基础部分的语法相比&#xff0c;ES6的语法进行了一些更加严谨的约束和优化&#…

OpenCV 图像基本操作

OpenCV快速通关 第一章&#xff1a;OpenCV 简介与环境搭建 第二章&#xff1a;OpenCV 图像基本操作 OpenCV 图像基本操作 OpenCV快速通关第二章&#xff1a;OpenCV 图像基本操作一、相关结构体与函数介绍&#xff08;一&#xff09;cv::Mat 结构体&#xff08;二&#xff09;c…

故障识别 | GADF-CNN-SSA-XGBoost数据分类预测/故障识别(Matlab)

故障识别 | GADF-CNN-SSA-XGBoost数据分类预测/故障识别&#xff08;Matlab&#xff09; 目录 故障识别 | GADF-CNN-SSA-XGBoost数据分类预测/故障识别&#xff08;Matlab&#xff09;分类效果基本描述程序设计参考资料 分类效果 基本描述 格拉姆角场差&#xff08;GADF&#…

树莓派4B android 系统添加led灯 Hal 层

本文内容需要用到我上一篇文章做的驱动&#xff0c;可以先看文章https://blog.csdn.net/ange_li/article/details/136759249 一、Hal 层的实现 1.Hal 层的实现一般放在 vendor 目录下&#xff0c;我们在 vendor 目录下创建如下的目录 aosp/vendor/arpi/hardware/interfaces/…

基于Matlab的变压器仿真模型建模方法(9):三相变压器组的建模仿真(续)

1.引言 前一节介绍了基于混合磁链的考虑主磁路饱和情况的三相变压器组的Simulink仿真模型,并应用它对Y,yn接法三相变压器组的瞬态过程进行了仿真计算。这一节建立一个新的基于等效电路的Simulink三相变压器组的仿真模型,并应用它对Y,d11接法的三相变压器组的瞬态过程进行仿…

【Atcoder】【ABC383】A- Humidifier 1加湿器 题解

前言 不知道大家有没有关注过AtCoder 这是小日子那边的一个网站&#xff0c;每周都会有比赛 比起CF等等&#xff0c;最大的优点就是延迟低&#xff0c;题目质量也不错 计划以后每周更新题解了 正文 题目传送门A - Humidifier 1 题目大意 有一个加湿器&#xff0c;给定 …

LeetCode 热题 100_反转链表(23_206_简单_C++)(单链表_递归)

LeetCode 热题 100_反转链表&#xff08;23_206&#xff09; 题目描述&#xff1a;输入输出样例&#xff1a;题解&#xff1a;解题思路&#xff1a;思路一&#xff08;迭代&#xff09;&#xff1a;思路二&#xff08;简化方法一(迭代)代码&#xff09;&#xff1a;思路三&…

49 基于单片机的湿度和光照监测

目录 一、主要功能 二、硬件资源 三、程序编程 四、实现现象 一、主要功能 基于AT89C52单片机&#xff0c;采用DHT11温湿度传感器检测土壤湿度&#xff0c;光敏电阻连接ADC0832数模转换器作为光敏传感器&#xff0c;然后通过LCD1602显示湿度和光照值&#xff0c;如果湿度低…

【C语言】程序设计--算法

文章目录 1. 判断两个数的大小并交换2. 计算三角形面积3. 根据x的值计算y4. 字符大小写转换5. 百钱百鸡问题6. 计算公式y的值7. 输出所有的水仙花数8. 计算n的阶乘9. 下三角数据10. 斐波那契数列11. 学生成绩统计12. 数组的平均值1. 判断两个数的大小并交换 介绍: 从键盘输入…

嵌入式Linux,字符串的处理,以及相关函数详解

C 语言库函数中已经给我们提供了丰富的字符串处理相关函数&#xff0c;基本常见的字符串处理需求都可 以直接使用这些库函数来实现。 1. 字符串输入/输出 在程序当中&#xff0c;经常需要在程序运行过程中打印出一些信息&#xff0c;譬如调试信息、报错信息、中间产生的变量的…

400G智算网络助力知名自动驾驶企业算力训练提效

根据Gartner的最新趋势预测&#xff0c;自动驾驶技术正迅速发展&#xff0c;预计在未来几年内将带来显著的商业效益&#xff0c;特别是在决策智能和边缘人工智能领域。目前&#xff0c;一家领军企业正积极拥抱基于大模型的数字化转型之路&#xff0c;作为自动驾驶领域的佼佼者&…

【iOS】OC高级编程 iOS多线程与内存管理阅读笔记——自动引用计数(三)

目录 ARC规则 概要 所有权修饰符 __strong修饰符 __weak修饰符 __unsafe_unretained修饰符 ___autoreleasing修饰符 ARC规则 概要 “引用计数式内存管理”的本质部分在ARC中并没有改变&#xff0c;ARC只是自动地帮助我们处理“引用计数”的相关部分。 在编译单位上可以…

数学活动是什么过程?

有专家说&#xff1a;数学活动是建构的操作过程&#xff0c;建构的过程必须是探索、发现和创造的过程。 什么是建构&#xff0c;建构就是构建&#xff0c;就是建立。明明有让人一看就明白的词&#xff0c;人非得弄得云遮雾绕。 也难怪&#xff0c;现在什么都流行上云。 上云…

windows11安装Linux子系统配置大数据hadoop

zai 1、安装linux子系统 1、启用适用于 Linux 的 Windows 子系统 搜索框里面输入<开发>即可跳转&#xff0c;打开开发人员模式 命令行里面输入systeminfo确定是否电脑已经支持虚拟化&#xff0c;是则可以继续安装: 2、然后先启用“适用于 Linux 的 Windows 子系统”可选…

RPC设计--从reactor设计 (IOthread)

主从reactor架构 一般的一个网络IO库都是主从reactor模式&#xff0c;即主线程中有一个MainReactor&#xff0c;其负责监听ListenFd&#xff0c;当接受到新的用户连接时&#xff0c;返回的clientfd并不会加入的MainReacotr&#xff0c;而是在子线程&#xff08;这里称为IO线程&…

微信创建小程序码 - 数量不受限制

获取小程序码&#xff1a;小程序码为圆图&#xff0c;且不受数量限制。 目录 文档 接口地址 请求方式 功能描述 注意事项 获取 scene 值 请求参数 返回参数 对接 请求方法 获取小程序码 调用获取小程序码 总结 文档 接口地址 https://api.weixin.qq.com/wxa/get…

【机器学习】基于SVM、逻辑回归和CNN的手写数字识别:性能对比与应用分析

基于SVM、逻辑回归和CNN的手写数字识别&#xff1a;性能对比与应用分析 1 基于SVM对手写数字识别2 基于逻辑回归对手写数字进行识别3 基于CNN对手写数字进行识别总结对比分析 1 基于SVM对手写数字识别 在使用SVM方法对手写数字进行识别的时候&#xff0c;我采用了一对多&#…

群控系统服务端开发模式-应用开发-邮件工厂电信189发送开发

一、电信189邮件工厂开发 1、添加框架对应的SDK composer require phpmailer/phpmailer 2、添加电信189邮件工厂 在根目录下extend文件夹下Mail文件夹下channel文件夹下&#xff0c;创建电信189邮件发送工厂并命名为DianxinMailSender。记住&#xff0c;一定要在电信189邮件发…

部署loki,grafana 以及springcloud用法举例

文章目录 场景docker 部署grafanadocker-compose部署loki维护配置文件 local-config.yaml维护docker-compose.yml配置启动 grafana 添加loki数据源springcloud用法举例查看loki的explore,查看日志 场景 小公司缺少运维岗位&#xff0c;需要研发自己部署日志系统&#xff0c;elk…

非常简单实用的前后端分离项目-仓库管理系统(Springboot+Vue)part 4

三十三、出入库管理 Header.vue导一下,RecordController加一个 //将入库数据和原有数据相加吧//新增PostMapping("/save")public Result save(RequestBody Record record) {return recordService.save(record) ? Result.success() : Result.fail();} GoodsManage.v…