【Linux 】文件描述符fd、重定向、缓冲区(超详解)

embedded/2024/10/10 18:54:28/

目录

​编辑 系统接口进行文件访问

open 接口介绍

文件描述符fd

重定向

缓冲区

1、缓冲区是什么?

2、为什么要有缓冲区?

3、怎么办?


我们先来复习一下,c语言对文件的操作:

C默认会打开三个输入输出流,分别是stdin, stdout, stderr

  1 #include<stdio.h>2 3 int main()4 {5     FILE *fp=fopen("log.txt","w");6     if(fp==NULL)7     {8         perror("error!");9         return 1;10     }11     fprintf(fp,"hello,%d,%s\n",10,"sxh");12     fclose(fp);13     return 0;14 }

 系统接口进行文件访问

操作文件,除了上述C接口(当然,C++也有接口,其他语言也有),我们还可以采用系统接口来进行文件访问;

  #include<stdio.h>#include<string.h>#include<sys/types.h>#include<sys/stat.h>#include<fcntl.h>#include<unistd.h>int main(){umask(0);int fd =open("log.txt",O_WRONLY | O_CREAT |O_TRUNC,0666);if(fd<0){   perror("open");return 1;}//const char *message ="hello Linux!\n";const char * message ="aaaa\n";write(fd,message,strlen(message));close(fd);}

open("log.txt",O_WRONLY | O_CREAT |O_TRUNC,0666);:以写方式打开,不存在就创建,存在就先清空!相当于c语言中的  “w”  方式;

open 接口介绍

open man 2 open

       #includ e <sys/types.h>
       #include <sys/stat.h>
       #include <fcntl.h>

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

pathname:要打开或创建的目标文件

flags: 打开文件时,可以传入多个参数选项,用下面的一个或者多个常量进行“或”运算,构成;  常见的参数有:

                        O_RDONLY: 只读打开

                        O_WRONLY: 只写打开
                        O_CREAT : 若文件不存在,则创建它。需要使用mode选项,来指明新文件的访问权限
                        O_APPEND: 追加写

mode: mode 只有 当 在 flags 中使用 O_CREAT 时才有效 , 否则被忽略.

返回值
                成功:新打开的文件描述符
                失败:-1

 

文件描述符fd

我们通过系统调用I/O打开一个文件的返回值,就是文件描述符:

int fd = open("log.txt",O_WRONLY | O_CREAT |O_TRUNC,0666);

这里fd就是log.txt的文件描述符;

如果我们同时打开三个文件,并把这三个文件的文件描述符打印出来,会发现:

文件描述符是从3开始的,那0、1、2呢?

Linux进程默认情况下会有3个缺省打开的文件描述符:分别是标准输入0, 标准输出1, 标准错误2.
0,1,2对应的物理设备一般是:键盘,显示器,显示器

根据上述的图,我们可以知道:

open干什么?

1、创建file

2、开辟文件缓冲区的空间,加载文件数据(延后)

3、查看进程的文件描述符表

4、file地址,填入对应的表下标中

5、返回下标

重定向

看一下下面的代码:

1 #include<stdio.h>2 #include<sys/types.h>3 #include<sys/stat.h>4 #include<fcntl.h>5 #include<unistd.h>6 #include<string.h>7 #include<stdlib.h>8 const char *filename="log.txt";9 10 int main()11 {12     //close(0);13     close(1);14     //close(2);15     int fd =open(filename,O_CREAT | O_WRONLY | O_TRUNC,0666);16     if(fd<0)17     {18         perror("open");19         return 1;20     }21     printf("printf,fd:%d\n",fd);22     fprintf(stdout,"fprintf,fd:%d\n",fd);23 24     fflush(stdout);25 26 27     close(fd);28     return 0;29 30 }

先看现象:

1、如果我们关闭的是0:打印出来的log.txt的文件描述符是0;关闭的是2,则打印出来的文件描述符是2;关闭的是1,则不会显示出来;因为文件描述符1表示的是显示器,而我们把显示器关掉了自然打印不出来;

2、我们会在log.txt中打印出printf和fprintf的内容:

现象解释:

文件描述符规则:查看自己的文件描述表,分配最小的没有使用的文件描述符;

由于我们把显示器给关掉了,那么文件描述符1就是最小的没有使用的,这样log.txt就会被分配1为文件描述符;这其实就是一个重定向操作;

看下面的图:

一开始fd_array[1]指向的是显示器,由于close(1),加上open log.txt这样fd_array[1]指向的就是log.txt;

因为重定向的本质是在内核中改变文件描述符特点下标的内容,与上层无关;

fprintf()/printf()  --->stdout  --->struct FILE --->stdout ---->_fileno ==1;

所以fprintf()/printf() 看的就是_fileno ==1;由于重定向后log.txt的fd为1 ,自然fprintf()/printf()的内容打印到log.txt中;

如果我们一直用这种手动重定向的方式,比较麻烦;可以用系统调用:dup2()

看一下代码:

  1 #include<stdio.h>2 #include<sys/types.h>3 #include<sys/stat.h>4 #include<fcntl.h>5 #include<unistd.h>6 #include<string.h>7 #include<stdlib.h>8 const char *filename="log.txt";9 10 int main()11 {12     int fd = open(filename,O_CREAT | O_WRONLY | O_TRUNC,0666);13     //dup2(fd,1);14 15     printf("hello world,fd:%d\n",fd);16     fprintf(stdout,"hello world\n");17     fflush(stdout);18     close(fd);19     return 0;20 }

上述的代码运行结果:会直接打印到显示器上;

如果把dup2(fd,1);复原:显示器上不会显示,而是在log.txt上显示

 其实本质就是将fd覆盖1 指向下标的内容;

缓冲区

1、缓冲区是什么?

缓冲区就是一段内存空间


2、为什么要有缓冲区?

给上层提供高效的IO体验,间接提高整体效率


3、怎么办?

     a、刷新策略
               1、立即刷新  fflush(stdout)(c语言的刷新)、fsync(fd)(系统调用的刷新)
               2、行刷新  (显示器)
               3、全缓冲  (缓冲区写满,才刷新,普通文件)
     b、特殊情况
               进程退出,系统会自动刷新
               强制刷新 


http://www.ppmy.cn/embedded/125495.html

相关文章

区块链技术在金融行业的应用与未来发展趋势

区块链技术作为近年来金融科技中的核心创新&#xff0c;正在深刻影响全球金融行业的运行方式。香港作为国际金融中心&#xff0c;在区块链技术的应用与创新方面扮演了重要角色。本文将深入探讨区块链技术在金融行业中的应用&#xff0c;并展望其未来发展趋势。 #### 一、区块链…

运用MinIO技术服务器实现文件上传——在Linux系统上安装和启动(一)

# MinIO 单机版环境搭建详解 ## 1. 简介 随着大数据时代的到来&#xff0c;数据存储的需求日益增大&#xff0c;如何有效地存储和管理大规模的非结构化数据成为许多企业和开发者面临的挑战。MinIO 作为一个高性能、分布式对象存储系统&#xff0c;致力于为用户提供简单、快速…

Qt和c++面试集合

目录 Qt面试 什么是信号&#xff08;Signal&#xff09;和槽&#xff08;Slot&#xff09;&#xff1f; 什么是Meta-Object系统&#xff1f; 什么是Qt的MVC模式&#xff1f; 1. QT中connect函数的第五个参数是什么&#xff1f;有什么作用&#xff1f; 3. 在QT中&#xff…

实验室装修:一系列设计方案需考虑哪些因素?

实验室装修设计看似简单&#xff0c;但其实背后藏着无数细节与思考。一个成功的实验室不仅需要满足科研的需求&#xff0c;还要确保在安全、环保和未来扩展性等方面都有良好的规划。今天&#xff0c;我们就来聊聊&#xff0c;实验室装修设计方案究竟需要考虑哪些因素。 一、实验…

Day06-数据库服务存储引擎

Day-06-数据库存储引擎知识 1、数据库联合索引应用2、数据库索引扩展信息&#xff08;扩展列信息说明&#xff09;3、数据库索引应用总结4、数据库存储引擎概述5、数据库存储引擎种类6、数据库存储引擎特性&#xff08;Innodb&#xff09;Innodb vs MyISAM7、数据库存储引擎应用…

使用pandas和numpy进行数据理解与清晰

处理缺失值 可以使用dropna()方法删除包含空值的行或列&#xff0c;或者使用fillna()方法填充空值。例如&#xff0c;可以用均值、中位数或众数来填充空值&#xff0c;代码示例如下&#xff1a; import pandas as pd df pd.read_csv(property-data.csv) df.fillna(df.mean(),…

C语言二级考试上机题

1.表示关系x≥y≥z应使用C语言表达式() A)(x>y)&&(y>z)B)(x>y)AND(y>z)C)x>y>zD)(x>y)&(y>z) 2.C语言可执行程序的开始执行点是() A)程序中第一条可执行语言B)程序中第一个函数 C)程序中的main函数D)包含文件中的第一个函数 3.以下if语…

PyTorch实现卷积神经网络CNN

一、卷积神经网络CNN 二、代码实现&#xff08;PyTorch&#xff09; 1. 导入依赖库 import torch from torch import nn, optim from torchvision import datasets, transforms from torch.utils.data import DataLoader nn&#xff1a;包含了torch已经准备好的层&#xff0c…