文件基础和文件fd

文章目录

正文开始前给大家推荐个网站,前些天发现了一个巨牛的 人工智能 学习网站, 通俗易懂,风趣幽默 ,忍不住分享一下给大家。 点击跳转到网站

预备知识

我们平时说文件就是说文件里面有什么,那么空文件有大小吗?答案肯定是有的,我们平时所说的文件除了文件的内容以外还存在文件的创建时间,大小等等,这是文件的属性,所以文件 = 内容 + 属性的。所以我们对文件的操作无非就是对文件内容或者对文件属性的操作。不管是内容还是属性,它们本质都是数据,所以他们被存储在磁盘中,就是磁盘文件。所以我们要访问一个文件的时候都是要把这个文件打开的,该文件被打开前就是磁盘文件,被打开后,因为CPU只和内存打交道,所以被打开的文件一定会被加载到内存。所以加载磁盘文件,一定会访问外设,这部分工作是由OS来做的。

一个进程是可以打开很多文件的,所以进程和文件的比例关系一定是 1 :n 。而在我们的系统中是有很多进程的,所以被打开的文件也可能会非常的多,所以OS要对这些文件进行管理,管理的本质就是先描述在组织,因此在OS中存在struct file结构体来描述文件,所以以后打开一个文件就创建一个file结构体,把他们用链表链起来,对文件的管理就成了对该链表的增删查改。

C语言的文件接口

C语言打开文件的是fopen

第一个参数就是打开文件的名称,可以使用绝对路径也可以使用相对路径,第二个参数是打开的模式,经常用的就是w以写的方式打开,但是每次打开都会清空文件,不存在就创建,还有一种常用的是a以追加的方式打开,每次打开不会清空文件,直接在文件结尾进行写入,依然是不存在就创建。还有r方式,是以只读的方式打开。经常用的就是这三种模式。

我们会发现w模式和我们指令所讲的输出重定向非常像。

a模式和追加重定向非常相似。

系统调用

我们再来认识一个系统调用open。

open函数是一个系统用调用,它的第一个参数就是文件名,和fopen一样,但是第二个参数是标志位。标志位有很多但是这里我们只介绍常用的几种。第三个参数是文件的权限,一般来说只有创建文件的时候需要设置。

关于函数传入标志为的技巧是Linux中常用的传参方式,接下来给兄弟们演示一下什么叫做标志位传参。

c 复制代码
#include <stdio.h>

#define Print1 1
#define Print2 (1 << 1)
#define Print3 (1 << 2)
#define Print4 (1 << 3)

void printflag (int flag)
{
    if(flag & Print1) printf("i am Print1\n");
    if(flag & Print2) printf("i am Print2\n");
    if(flag & Print3) printf("i am Print3\n");
    if(flag & Print4) printf("i am Print4\n");
}
int main()
{
    printflag(Print1);
    printf("============================\n");
    printflag(Print1 | Print2);
    printf("============================\n");
    printflag(Print1 | Print2 | Print3);
    printf("============================\n");
    printflag(Print1 | Print2 | Print3 | Print4);
    printf("============================\n");
    return 0;
}

所以open的第二个参数第原理和这个基本上是差不多的,它的选项常用的O_WRONLY(只写)、O_RDONLY(只读)、O_CREAT(不存在就创建)、O_TRUNC(每次打开时清空文件)、O_APPEND(追加写,不清空文件)。open的返回值是一个fd(文件描述符),它是用来表示一个文件的。所以C语言中的FILE也一定封装了这个数字。有了这些选项的基础,我们可以来模仿实现一下fopen的各个选项的实现。

c 复制代码
FILE _fopen(const char * str, char c)
{
    int flag = 0;
    int is_read;
    if(c == 'a')
    {
        flag = O_WRONLY | O_APPEND | O_CREAT;
    }
    else
    {
        if(c == 'w')
        {
        flag = O_WRONLY | O_TRUNC | O_CREAT;

        }
        else
        {
            if(c == 'r')
            {
                flag = O_RDONLY;
                is_read = 1;
            }
            else
            {
                //TODO
            }
        }
    }

    int fd = 0;
    if(is_read)
    {
        fd = open(str, flag, 0x666);
        if(fd < 0)
        {
            perror("open");
            exit(-1);
        } 
    }
    else
    {
        fd = open(str, flag);
        if(fd < 0)
        {
            perror("open");
            exit(-1);
        } 
    }
    FILE file;
    // _fileno就是文件描述符
    file._fileno = fd;
    return file;
}

所以C语言的所有库函数的本质都是封装了系统调用。

文件fd

到这里我们可以来尝试理解一下文件了。如何在系统层面上理解一下文件呢?

我们知道每个进程在被创建是都是会有自己的PCB的,在Linux中也就是task_struct,所以每个进程的PCB中都有一个struct files_struct* files 的指针,这个指针指向的结构体中有一个非常重要的一张表,struct file* fd_array[],这是一个指针数组,我们打开的每一个文件都会被在这个指针数组中被指向,一般来说是从小到大来排列的,而数组的下标就是我们上面系统调用返回的文件描述符。所以文件描述符的本质就是数组的下标。操作系统访问文件只认识文件描述符。

我们进程在运行的时候,是会默认打开三个流,标准输入流、标准输出流、标准错误流。这三个流对应的硬件分别是键盘、显示器、显示器。因为Linux下一切皆文件,所以这三个流在进程被打开时会一次把文件描述符表的0、1、2位置给占了,所以我们自己打开的文件的fd一般是从3开始从小到大排的。

OS默认打开三个流,就是为了我们程序员默认进行输入输出的代码的编写。

我们如何理解一切接文件?

在file文件中是有函数指针的,所以对于不同的文件我们让它的文件指针指向对应的方法,如果没有改方法的话我们让这个指针指向空就行了,所以在上层看来,文件就是这个方法,但是它是可能对于不同的文件指向的方法也是不同的。

相关推荐
暗碳38 分钟前
华为麦芒5(安卓6)termux记录 使用ddns-go,alist
android·linux
靡樊39 分钟前
Linux:基础IO
linux
菜鸟康1 小时前
Linux系统编程——理解系统内核中的信号捕获
linux·运维·服务器
张明奇-琦玉1 小时前
Boost之log日志使用
linux·服务器·算法
云计算DevOps-韩老师2 小时前
【网络云计算】2024第52周-每日【2024/12/26】小测-理论&实操-备份MySQL数据库并发送邮件-解析
linux·开发语言·网络·数据库·mysql·云计算·perl
ac.char2 小时前
Ubuntu系统下 npm install -g tauri 报错问题处理
linux·ubuntu·npm
无名3873 小时前
Debian12 安装配置 ODBC for GaussDB
linux·运维·gaussdb
小Mie不吃饭3 小时前
Linux | 零基础Ubuntu解压RaR等压缩包文件
linux·运维·ubuntu
drebander3 小时前
SQL 实战:日期与时间函数 – 统计数据的时间跨度与趋势
linux·数据库·sql
年纪青青4 小时前
泰山派GPIO子系统驱动---亮灯
linux·泰山派·gpio驱动