【Linux文件管理】重定向&&内核级缓冲区&&用户级缓冲区
发布时间 - 2025-04-17 00:00:00 点击率:次文件管理文件描述符表(files_struct)
上一期我们将文件描述符讲完了,这期来讲讲文件管理中的文件描述符表,在task_struct有一个指针是指向文件描述符表的。
我们进入进入文件描述符表:
可以看到当中有很多属性,虽然这些我们都不知道,但是有一个我么是知道的,就是红框框起来的,这个我们是知道的,这个是一个文件的指针数组,这个数组的下标就是文件描述符。 我们来画一个形象的结构。
文件描述符表的结构如图所示,我们将其余属性给屏蔽了,只留下一个指向文件的指针数组,这个file的指针数组,内存存储的是文件的属性,不仅是大小和路径还有很多属性。 有了文件描述符表,进程和文件就可以通过文件描述符表来管理了。 我们先简单写一段代码:
代码语言:javascript代码运行次数:0运行复制#include#include #include #include #include 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,为什么呢?
通过这个示例我们可以画出下面的图:
前三个是运行程序的时候默认打开的三个流,假如我们任意关闭一个会出现什么状况呢,我们打开的新的文件会去占据腾出来的那个空位吗? 代码:
代码语言:javascript代码运行次数:0运行复制#include#include #include #include #include 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的位置。
我们试试关闭输出流:
当我们关闭输出流的时候屏幕上是不会打印的,因为我们将输出流给关闭了,所以不会在屏幕上打印,又因为我们打开的文件占据了以前输出流数组下标对应的位置,所以不会打印在屏幕上,会打印在文件中,我们来查看一下:
可以看见文件中也没有,这里先不解释,这里其实存在一个用户级缓冲区,还没刷新到内核缓冲区当中,就将文件关闭了,所以这里不会写入到文件当中。 这里只需要强制刷新一下即可:
代码语言:javascript代码运行次数:0运行复制#include#include #include #include #include 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指向。 我们写一段简单的代码:
代码语言:javascript代码运行次数:0运行复制#include#include #include #include #include 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 操作时都直接与操作系统或外设进行交互,从而降低性能损耗。 回到之前的问题:
代码语言:javascript代码运行次数:0运行复制#include#include #include #include #include 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,程序退出前会自动刷新。 为什么会存在用户级缓冲区呢?
从用户级缓冲区强制刷新到内核级缓冲区叫fflush,从内核级缓冲区强制刷新到外设当中叫fsync fsync:
通过本文对文件管理及其内核级与用户级缓冲区的详细探讨,我们对操作系统中文件管理的机制有了更深刻的理解。首先,我们了解了 files_struct 的作用及其如何在内核中管理文件描述符的详细实现,掌握了文件的重定向以及文件描述符的相关操作。其次,通过对内核级缓冲区与用户级缓冲区的对比分析,我们认识到这两者在性能优化和内存管理中的关键作用,特别是在提升 I/O 性能和减少系统调用开销方面的重要性。
文件管理作为操作系统中非常核心的部分,不仅直接影响着系统资源的利用效率,也对程序的执行性能和稳定性起着至关重要的作用。通过合理管理内核级缓冲区和用户级缓冲区,操作系统能够有效地平衡性能和资源消耗,在确保数据准确性的同时提升系统的响应速度和吞吐量。
总之,深入理解操作系统中的文件管理机制,能够帮助我们更好地设计和优化应用程序,提升系统的整体效率。未来,随着操作系统的不断发展和优化,文件管理机制将会面临更多新的挑战和机遇,我们也将继续探索和学习,以应对新的技术发展。
# linux
# 操作系统
# ai
# 为什么
# JavaScript
# fclose
# printf
# 字符串
# 指针
# 性能优化
# unix
# 重定向
# 外设
# 文件管理
# 的是
# 屏幕上
# 还没有
# 可以看见
# 应用程序
# 这就是
相关栏目:
【
网站优化151355 】
【
网络推广146373 】
【
网络技术251813 】
【
AI营销90571 】
相关推荐:
Laravel如何与Docker(Sail)协同开发?(环境搭建教程)
Android自定义listview布局实现上拉加载下拉刷新功能
如何挑选最适合建站的高性能VPS主机?
宙斯浏览器文件分类查看教程 快速筛选视频文档与图片方法
绝密ChatGPT指令:手把手教你生成HR无法拒绝的求职信
如何在阿里云通过域名搭建网站?
jquery插件bootstrapValidator表单验证详解
Laravel如何保护应用免受CSRF攻击?(原理和示例)
Laravel如何理解并使用服务容器(Service Container)_Laravel依赖注入与容器绑定说明
Laravel Octane如何提升性能_使用Laravel Octane加速你的应用
php后缀怎么变mp4格式错误_修改扩展名提示格式不对怎么办【技巧】
jQuery中的100个技巧汇总
Python制作简易注册登录系统
如何挑选高效建站主机与优质域名?
Laravel路由怎么定义_Laravel核心路由系统完全入门指南
java中使用zxing批量生成二维码立牌
如何在腾讯云服务器上快速搭建个人网站?
Laravel定时任务怎么设置_Laravel Crontab调度器配置
济南网站建设制作公司,室内设计网站一般都有哪些功能?
javascript中的try catch异常捕获机制用法分析
INTERNET浏览器怎样恢复关闭标签页_INTERNET浏览器标签恢复快捷键与方法【指南】
WordPress 子目录安装中正确处理脚本路径的完整指南
Laravel Eloquent性能优化技巧_Laravel N+1查询问题解决
网站制作企业,网站的banner和导航栏是指什么?
Laravel如何与Pusher实现实时通信?(WebSocket示例)
LinuxShell函数封装方法_脚本复用设计思路【教程】
Laravel怎么做缓存_Laravel Cache系统提升应用速度的策略与技巧
北京企业网站设计制作公司,北京铁路集团官方网站?
Laravel怎么写单元测试_PHPUnit在Laravel项目中的基础测试入门
家族网站制作贴纸教程视频,用豆子做粘帖画怎么制作?
IOS倒计时设置UIButton标题title的抖动问题
如何在企业微信快速生成手机电脑官网?
智能起名网站制作软件有哪些,制作logo的软件?
php 三元运算符实例详细介绍
详解免费开源的DotNet二维码操作组件ThoughtWorks.QRCode(.NET组件介绍之四)
Android中AutoCompleteTextView自动提示
Laravel如何记录自定义日志?(Log频道配置)
实例解析Array和String方法
如何在IIS7中新建站点?详细步骤解析
Java Adapter 适配器模式(类适配器,对象适配器)优缺点对比
MySQL查询结果复制到新表的方法(更新、插入)
网站视频制作书签怎么做,ie浏览器怎么将网站固定在书签工具栏?
Laravel如何配置中间件Middleware_Laravel自定义中间件拦截请求与权限校验【步骤】
javascript中闭包概念与用法深入理解
公司门户网站制作流程,华为官网怎么做?
Midjourney怎样加参数调细节_Midjourney参数调整技巧【指南】
如何安全更换建站之星模板并保留数据?
Win11摄像头无法使用怎么办_Win11相机隐私权限开启教程【详解】
Laravel怎么进行数据库回滚_Laravel Migration数据库版本控制与回滚操作
如何实现javascript表单验证_正则表达式有哪些实用技巧


fileno); printf("stdout:fileno:%d\n",stdout->_fileno); printf("stderr:fileno:%d\n",stderr->_fileno); close(fd); return 0;}