【Linux】C文件系统详解(二)——什么是fd文件描述符以及理解“一切皆文件“

news2024/12/23 5:18:56

文章目录

  • fd-文件描述符
  • 如何深度理解"一切皆文件"
    • **我们使用OS的本质:**
    • FILE
      • `FILE`是什么?谁提供的?和我们刚刚讲的内核的struct有关系吗
        • `FILE`是一个结构体.该结构体内部一定要有以下字段:
        • `FILE`是C语言标准库提供的.
        • `FILE`和我们刚刚讲的内核的struct没有关系,最多就是上下层的关系
    • 做实验->重定向的本质
      • 第一个实验->文件描述符的分配规则
      • 第二个实验->输出重定向
        • 重定向的原理
      • 第三个实验->输入重定向
      • 第四个实验->追加重定向
      • 结论
        • 需求:把常规消息放一个文件,错误消息放在另一个文件
        • 更好的写法
    • 让我们自己的程序支持重定向:

fd-文件描述符

任何一个进程,在启动的时候,默认会打开当前进程的三个文件

标准输入标准输出标准错误本质都是文件
stdinstdoutstderr文件在语言层的表现
cincoutcerr同上,但是他是一个类
012<-fd ,数组下标

文件描述符,即open对应的返回值,本质就是:数组下标
标准输出和标准错误都会向显示器打印,但是其实是不一样的

类型设备文件
标准输入键盘文件
标准输出显示器文件
标准错误显示器文件
#include<iostream>
#include<cstdio>

int main()
{
	//因为linux一切皆文件,所以,向显示器打印,本质就是向文件中写入
	printf("hello printf->stdout\n");	
	fprintf(stdout,"hello fprintf->stdout\n");
	fprintf(stderr,"hello fprintf->stderr\n");

	std::cout << "hello cout -> cout" << std::endl;
	std::cerr << "hello cerr -> cerr" << std::endl;
}
int fd1 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);//3
int fd2 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);//4
int fd3 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);//5
int fd4 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);//6
int fd5 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);//7
int fd6 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);//8

![[Drawing 2023-11-08 11.12.05.excalidraw|900]]

如何深度理解"一切皆文件"

我们使用OS的本质:

都是通过进程的方式进行操作系统的访问,在进程的角度,只能看到文件对象,而看不到底层的设备的区别,所以我们才说"Linux下一切皆文件".
![[文件系统 2023-03-18 15.19.03.excalidraw|800]]

FILE

操作系统层面,我们必须使用fd才能找到文件!
任何语言层面访问外设或者文件,都必须经历OS

FILE是什么?谁提供的?和我们刚刚讲的内核的struct有关系吗

#include<stdio.h>
FILE* fopen(const char *path,const char* mode);

答案:

FILE是一个结构体.该结构体内部一定要有以下字段:
fd

证明:

int main()
{
	printf("%d\n",stdin->_fileno);
	printf("%d\n",stdout->_fileno);
	printf("%d\n",stderr->_fileno);
	FILE* fp = fopen(LOG,"w");
	printf("%d\n",fp->_fileno);
}
FILE是C语言标准库提供的.

我们平时安装VS2019,是在安装IDE环境以及对应语言的库和头文件

FILE和我们刚刚讲的内核的struct没有关系,最多就是上下层的关系

做实验->重定向的本质

第一个实验->文件描述符的分配规则

把fd为3的文件关闭以后,新的文件fd应该是什么

int main()
{
	close(0);//fclose(stdin)
	close(2);//fclose(stderr)
	
	int fd1 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);
	int fd2 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);
	int fd3 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);
	int fd4 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);
	int fd5 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);
	int fd6 = open(LOG,O_WRONLY | O_CREAT | O_TRUNC,0666);

	printf("%d\n",fd1);0
	printf("%d\n",fd2);2
	printf("%d\n",fd3);3
	printf("%d\n",fd4);4
	printf("%d\n",fd5);5

	return 0;
}

进程中,文件描述符的分配规则:

最小的,没有被使用的数组元素,分配给新文件

第二个实验->输出重定向

int main()
{
	fclose(1);
	int fd = open(LOG, O_WRONLY | O_CREAT | O_TRUC, 0666);
	//此时log.txt的fd是'1'
	//但是上层结构不知道这个变化,他只知道要写进fd=1的文件中
	printf("you can see me!\n");//本来是指向stdout -> 1的,但是stdout变成了log.txt
	printf("you can see me!\n");
	printf("you can see me!\n");
	printf("you can see me!\n");
	printf("you can see me!\n");
	
	return 0;
}

结果:打印不到屏幕,但是打印到了log.txt
printf("",);不是认stdout,而是认fd==1的文件描述符

重定向的原理

在上层无法感知的情况下,在OS内部,更改进程对应的文件描述符表中,特定下标的指向!!

第三个实验->输入重定向

现在log.txt中写入:

123 456
int main()
{
	fclose(0);
	int fd = open(LOG, O_RDONLY | O_CREAT | O_TRUC, 0666);//fd=0
	int a,b;
	scanf("%d %d",&a,&b);
	printf("a=%d , b=%d\n",a,b);
	return 0;
}

结果: cat log.txt:
a=123 , b=456

第四个实验->追加重定向

int main()
{
	close(1);//标准输出
	int fd = open(LOG, O_RDONLY | O_CREAT | O_APPEND, 0666);
	printf("you can see me!\n");//从屏幕(stdout)上追加到fd中
	printf("you can see me!\n");
	printf("you can see me!\n");
	printf("you can see me!\n");
}

结果: cat log.txt:

a=123 , b=456
you can see me!
you can see me!
you can see me!
you can see me!

结论

所以stdout cout->1,他们都是向1号文件描述符对应的文件打印
stderr cerr ->2 ,他们都是向2号文件描述符对应的文件打印
输出重定向,只改的是1号对应的指向,对2号不影响

需求:把常规消息放一个文件,错误消息放在另一个文件
#include<stdio.h>
#include<sys/types.h>
#include<sys/stat.h>
#include<fcntl.h>
#include<unistd.h>
#include<errno.h>
#include<string.h>
#define LOG "log.txt"
#define LOG_NORMAL "logNormal.txt"
#define LOG_ERROR "logError.txt"

int main()
{
	close(1);
	int fd = open(LOG_NORMAL, O_WRONLY | O_CREAT | O_APPEND, 0666);
	close(2);
	int fd = open(LOG_ERROR, O_WRONLY | O_CREAT | O_APPEND, 0666);

	printf("hello printf->stdout\n");	
	fprintf(stdout,"hello fprintf->stdout\n");
	fprintf(stderr,"hello fprintf->stderr\n");

}

所以为什么要默认把1和2打开:
就是为了把常规消息和错误消息分类开来,便于后面的调试!

bash中重定向操作

a.out > log.txt 2 > &1
或者
a.out 1>log.txt 2>err.txt

2 > &1
把1里面的内容,写到2下标的内容里

更好的写法

int dup2(int oldfd, int newfd)
是对数组对应下标的内容进行拷贝

new要成为old的拷贝
所以最终只有oldfd的内容了
而我们最后正确重定向肯定是剩下3啊
所以oldfd 是3
newfd 是1

所以代码
dup2(fd,1)

重定向写法:

int main ()
{
	int fd = open(LOG_NORMAL, O_WRONLY | O_CREAT | O_APPEND, 0666);
	if(fd < 0)
	{
		perrer("open");
		return 1;
	}
	dup2(fd,1);
	printf("hello world,hello lx\n");
	close(fd);
}

就是打开文件,之后使用dup2就行

让我们自己的程序支持重定向:

enum redir{
	REDIR_INPUT = 0,
	REDIR_OUTPUT,
	REDIR_APPEND,
	REDIR_NONE
};

char* checkdir(char commandstr[],redir &redir_type);
{
	//1.监测是否有 > < >> 
	//2.如果有,要根据> < >> 设置redir_type = ?
	//3.将符号改成\0,分成两部分
	//保存文件名,并返回
	//如果不满足,直接返回
}

int main()
{
	while(1)
	{
			redir redir_type = REDIR_NONE
		//...
		char* filename = NULL;
		char* filename = checkdir(commandstr,&redir_type);
		if(*filename)
		{
			//1.存在文件
			//2.获取redir_type
		}
		//遍历,看是否有> < >>,这三个字符
		//前半部分执行后续
		//把这三个字符变成\0,将后面的字符串打开文件
		//dup2(fd,1);
	
	
		//...
		if(id == 0)
		{
			if(redir_typr != REDIR_NONE)
			{
				dup2();
			}
		}
	}
}

未完待续…

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

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

相关文章

医院绩效考核系统源码 医院绩效考核系统方案

医院绩效考核系统源码 医院绩效考核系统是现代医院管理的重要方法和科学的管理工具。良好的绩效管理&#xff0c;有助于带动全院职工的工作积极性&#xff0c;有助于提高工作效率、提高医疗质量、改善服务水平、降低运营成本&#xff0c;全面提升医院的精细化管理水平。 医院绩…

不允许你还没有了解哈希表、哈希桶、哈希冲突的解决,如何避免冲突

✏️✏️✏️今天给各位带来的是哈希桶、哈希冲突方面的知识。 清风的CSDN博客 &#x1f61b;&#x1f61b;&#x1f61b;希望我的文章能对你有所帮助&#xff0c;有不足的地方还请各位看官多多指教&#xff0c;大家一起学习交流&#xff01; 动动你们发财的小手&#xff0c;点…

springMVC学习笔记-请求映射,参数绑定,响应,restful,响应状态码,springMVC拦截器

目录 概述 springMVC做了什么 springMVC与struts2区别 springMVC整个流程是一个单向闭环 springMVC具体的处理流程 springMVC的组成部分 请求映射 RequestMapping 用法 属性 1.value 2.method GET方式和POST方式 概述 HTTP给GET和POST做了哪些规定 GET方式&…

IDEO也不行了吗?设计正在变革#实时设计

2023 年 8 月&#xff0c;在与宜家品牌合作近 10 年之后&#xff0c;SPACE10 关门了。 最近&#xff0c;IDEO&#xff0c;设计思维的早期倡导者和践行者&#xff0c;宣布裁员1/3。 介绍下这两家设计公司&#xff1a; SPACE10 由宜家全额资助&#xff0c;于 2015 年落户哥本哈根…

生成式AI模型量化简明教程

在不断发展的人工智能领域&#xff0c;生成式AI无疑已成为创新的基石。 这些先进的模型&#xff0c;无论是用于创作艺术、生成文本还是增强医学成像&#xff0c;都以产生非常逼真和创造性的输出而闻名。 然而&#xff0c;生成式AI的力量是有代价的—模型大小和计算要求。 随着生…

DevSeo Studio设置中文界面

安装好DevSeo Studio后默认打开是欢迎页。 左下角Configure点击展开&#xff0c;选择plugins 弹出页面选择“installed”,然后输入chinese,默认是关闭的&#xff0c;点击enable将它启用&#xff0c;然后点击OK。 弹出页面点击“restart”重启即可。

Draco Win10编译

1. 工具 CMake 3.2.8&#xff0c;Visual Studio 2019 2. 步骤 2.1 拷贝代码 git clone https://github.com/google/draco.gitgit clone https://github.com/google/draco.git 下载第三方依赖 git submodule sync git submodule update --init --recursive 2.2 CMake编译…

一个UE无法注册的问题

问题场景是环境中只有一个小区&#xff0c;UE在找到这个小区&#xff0c;收到MIB SIB1后一直不发起注册。我想这大概是和S准则不满足有关系了&#xff0c;这个问题基本是又没啥好看的了&#xff0c;太简单了&#xff0c;在SIB1周围找找就解决了&#xff0c;于是我发现了以下log…

Find My充电宝|苹果Find My技术与充电宝结合,智能防丢,全球定位

充电宝是一种个人可随身携带&#xff0c;自身能储备电能&#xff0c;主要为手持式移动设备等消费电子产品&#xff08;例如无线电话、笔记本电脑&#xff09;充电的便携充电器&#xff0c;特别应用在没有外部电源供应的场合。其主要组成部分包括&#xff1a;用作电能存储的电池…

从算法到应用:直播美颜滤镜SDK的全面解读与评测

直播美颜滤镜SDK技术逐渐成为直播平台不可或缺的一环。本文将对直播美颜滤镜SDK进行全面解读&#xff0c;深入探讨其算法原理和应用效果&#xff0c;并通过评测分析展现其在直播领域的实际价值。 一、算法原理解读 直播美颜滤镜的背后是复杂而精密的算法&#xff0c;旨在提升…

【docker】iptables实现NAT

iptables是一个Linux内核中的防火墙工具&#xff0c;可以被用来执行各种网络相关的任务&#xff0c;如过滤、NAT和端口转发等&#xff0c;可以监控、过滤和重定向网络流量。 iptables可以用于以下应用场景&#xff1a; 网络安全&#xff1a;iptables可以过滤网络流量&#xf…

吉林省土木建筑学会建筑电气分会及吉林省建筑电气情报网学术交流年会-安科瑞 蒋静

11月9-10日&#xff0c;吉林省土木建筑学会建筑电气分会及吉林省建筑电气情报网学术交流年会在吉林长春隆重举办。安科瑞电气股份有限公司作为智慧用电产品供应商受邀参会&#xff0c;为参会人士展示了安科瑞能源物联网云平台、电力运维云平台、智慧消防云平台、预付费管理云平…

22款奔驰S450L升级流星雨大灯 感受最高配的数字大灯

“流星雨”数字大灯&#xff0c;极具辨识度&#xff0c;通过260万像素的数字微镜技术&#xff0c;实现“流星雨”仪式感与高度精确的光束分布&#xff1b;在远光灯模式下&#xff0c;光束精准度更达之前84颗LED照明的100倍&#xff0c;更新增坡道照明功能&#xff0c;可根据导航…

[开源]基于 AI 大语言模型 API 实现的 AI 助手全套开源解决方案

原文&#xff1a;[开源]基于 AI 大语言模型 API 实现的 AI 助手全套开源解决方案 一飞开源&#xff0c;介绍创意、新奇、有趣、实用的开源应用、系统、软件、硬件及技术&#xff0c;一个探索、发现、分享、使用与互动交流的开源技术社区平台。致力于打造活力开源社区&#xff0…

java线性并发编程介绍-锁(二)

2.5 重量锁底层ObjectMonitor 需要去找到openjdk&#xff0c;在百度中直接搜索openjdk&#xff0c;第一个链接就是 找到ObjectMonitor的两个文件&#xff0c;hpp&#xff0c;cpp 先查看核心属性&#xff1a;http://hg.openjdk.java.net/jdk8u/jdk8u/hotspot/file/69087d08d473…

吴恩达《机器学习》9-1:代价函数

一、引入新标记方法 首先&#xff0c;引入一些新的标记方法&#xff0c;以便更好地讨论神经网络的代价函数。考虑神经网络的训练样本&#xff0c;其中每个样本包含输入 x 和输出信号 y。我们用 L 表示神经网络的层数&#xff0c;表示每层的神经元个数&#xff08;表示输出层神…

数据结构与算法面试题——C++

自己在秋招过程中遇到的数据结构与算法方面的面试题 数据结构 vector vector是⼀种序列式容器&#xff0c;与array唯⼀差别就是对于空间运⽤的灵活性 array占⽤的是静态空间&#xff0c;⼀旦配置了就不可以改变⼤⼩&#xff0c;如果遇到空间不⾜的情况还要⾃⾏创建更⼤的空间…

处理BOP数据集,将其和COCO数据集结合

处理BOP数据集&#xff0c;将其和COCO数据集结合 BOP 取消映射关系&#xff0c;并自增80 取消文件名的images前缀 import os import json from tqdm import tqdm import argparseparser argparse.ArgumentParser() parser.add_argument(--json_path, defaultH:/Dataset/COCO…

Ghidra逆向工具配置 MacOS 的启动台显示(Python)

写在前面 通过 ghidra 工具, 但是只能用命令行启动, 不太舒服, 写个脚本生成 MacOS 的 app 格式并导入启动台. 不算复杂, 主要是解析包的一些元信息还有裁剪软件图标(通过 MacOS 自带的 API) 脚本 #!/opt/homebrew/bin/python3import os import re import subprocess as sp…

《Linux从练气到飞升》No.31 多线程编程实践与线程安全技术

&#x1f57a;作者&#xff1a; 主页 我的专栏C语言从0到1探秘C数据结构从0到1探秘Linux菜鸟刷题集 &#x1f618;欢迎关注&#xff1a;&#x1f44d;点赞&#x1f64c;收藏✍️留言 &#x1f3c7;码字不易&#xff0c;你的&#x1f44d;点赞&#x1f64c;收藏❤️关注对我真的…