[Linux]文件描述符fd

1.什么叫当前路径

  • 进程在启动时,会自动记录自己启动时所在的路径,通过ls /proc/PID -l命令,查看cwd。
    在这里插入图片描述

  • 用chdir()接口可以更改进程的工作目录

2.程序默认打开的文件流

  • stdin(标准输入),stdout(标准输出),stderr(标准错误)。所以说在我们刚开始雪编程的时候,使用的printf(),scanf(),这些函数的时候,我们并没有打开键盘文件和显示器文件,但是我们仍然可以向键盘输入,从显示器输出,因为程序在打开的时候,默认已经帮我们打开了。
int main()
{
    printf("hello printf\n");
    fputs("hello fputs\n", stdout);
}
  • 如上代码所示,可以指定向stdout写入,并且stdout不需要声明,而printf函数,默认的输入对象就是stdout,可以不写。
  • 即stdin,stdout,stderr可以直接被使用!

3.访问文件不仅仅需要C语言上的文件接口,OS还必须提供对应的访问文件的系统调用

在这里插入图片描述

#define ONE 1
#define TWO (1<<1)
#define THREE (1<<2)
#define FOUR (1<<3)
#define FIVE (1<<4)
//print函数中只用一个标志位,可以标志多个状态
void Print(int flag)
{
    if(flag & ONE)printf("1\n");
    if(flag & TWO)printf("2\n");
    if(flag & THREE)printf("3\n");
    if(flag & FOUR)printf("4\n");
    if(flag & FIVE)printf("5\n");
}

int main()
{
    Print(ONE);
    Print(TWO);
    Print(ONE |TWO);
    Print(THREE|FOUR|FIVE);
    Print(ONE | TWO | THREE | FOUR | FIVE);
}
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>
int main()
{
    //系统掩码设为0,系统掩码默认是0002,创建的文件权限码需要减去系统掩码对应的1,即代数第二位置为0.
    umask(0);
    //O_WRONLY | O_CREAT 以写的形式打开文件,如果文件不存在的话,则创建。
    //0666当用open函数打开不存在的文件时,需要第三个参数,第三个参数的作用就是设置新建文件的权限
    //O_TRUNC代表如果文件存在的话,先把文件做清空处理。如果没有O_TRUNC,那么往文件写的时候,会在文件开头进行写,并且文件之前的内容还会在文件中,这样就会导致,新写入的数据会覆盖之前已经存在的数据。
    int fd = open("log.txt", O_WRONLY | O_CREAT | O_TRUNC, 0666);
    //O_APPEND代表每次写文件都是往文件尾部进行写入。
    // int fd = open("log.txt", O_WRONLY | O_CREAT | O_APPEND, 0666);
    if(fd =-1)
    {    
        perror("open");
        return 1;
    }
    const char* str = "hello Linux\n";
    //strlen(str)后面不需要加一,因为\0是不需要写入到文件里面的,\0只是在c语言中代表字符串的结束标志,与文件没有关系,如果写入的话,就成了文件中的乱码了。
    write(fd, str, strlen(str));
    close(fd);
    return 0;
}
#include <stdio.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <unistd.h>
#include <string.h>
int main()
{
    int fd1 = open("log.txt", O_WRONLY | O_CREAT | O_TRUNC, 0666);
    int fd2 = open("log.txt", O_WRONLY | O_CREAT | O_TRUNC, 0666); 
    int fd3 = open("log.txt", O_WRONLY | O_CREAT | O_TRUNC, 0666); 
    int fd4 = open("log.txt", O_WRONLY | O_CREAT | O_TRUNC, 0666); 
    int fd5 = open("log.txt", O_WRONLY | O_CREAT | O_TRUNC, 0666);

    printf("fd:%d\n",fd1);
    printf("fd:%d\n",fd2);
    printf("fd:%d\n",fd3);
    printf("fd:%d\n",fd4);
    printf("fd:%dIn",fd5);
    close(fd);
    return 0;
}
  • 上面这段代码的运行结果是在这里插入图片描述
    ,之所以从3开始,是因为0, 1, 2已经被用了,分别是标准输入,标准输出,标准错误。

  • 在这里插入图片描述

  • 如上图所示,在用户操作接口中,我们使用的fopen()函数,底层封装了open()这样的系统调用接口。

  • 那么为什么C语言要对linux进行封装,因为这样可以保证C语言的可移植性,保证他的跨平台性。

4.打开文件操作的底层实现

在这里插入图片描述

  • 如上图所示,打开一个文件
    1. 首先需要在磁盘中找到这个文件,这个文件包含文件内容和文件属性。
    1. 然后将属性和内容加载到内存当中,并创建struct file 结构体,在该结构体中初始化属性,方法和缓冲区。同时将该结构体放到打开文件列表中进行管理。
    1. 最后,将该结构体的地址传到进程管理task_struct中struct files_struct *file所指向的一个文件控制块指针数组中。
    1. 最后将该数组的下标返回给用户。
  • 综上所示,解释了open,write函数在使用的时候,第一个参数为什么是fd了,因为进程在执行open时,根据fd来找到对应的打开文件,首先在缓冲区看有没有数据,如果有,直接读到open函数指定的BUFF缓冲区,如果没有,根据打开文件来找到磁盘中的文件,再对其进行一系列处理。
  • 得到一个结论,文件描述符fd的本质就是数组下标。

5.解linux系统中一切皆文件

  • 在这里插入图片描述

  • 如上图所示,当上层用户要访问硬件时,只需要调用read函数,而read函数中已经封装了文件控制块中的函数指针,这个函数指针就可以调用设备。

6.文件描述符fd的分配规则

  • 最小的没有被分配的数组下标,会分配给最新的打开文件。如下代码所示,输出的fd就等于0.
int main()
{
    close(0);
    int fd = open("log.txt", O_RDONLY);
    printf("%d", fd);
    return 0;
}

7.利用规则实现重定向

int main()
{
    close(1);
    int fd = open("log.txt", O_WRONLY);
    printf("fd:%d", fd);
    return 0;
}
  • 上面这段代码的结果,按照我们之前的认识来讲的话,应该会在命令函中输出一个fd:1;但是运行这段代码后,并没有看到显示器上有任何的反应,但是当我们执行cat log.txt时,输出的是fd:1,也就是说这段代码往log.txt文件里面写了fd:1。
  • 对于这个现象,是因为close和open这两个是属于系统调用,将下标为1的位置显示器文件关闭了,取而代之的是log.txt文件,这是属于系统层面的,在C语言层面并不知晓,所以在执行printf函数时,默认向stdout中输出,stdout中有个变量_fileno = 1,也就是向下标为1的位置进行输出,也就是向更改后的log.txt进行输出。

8.系统提供重定向接口

int main()
{
    int fd = open("log.txt", O_WRONLY);
    dup2(fd, 1);
    printf("hello Linux!");
    return 0;
}
  • 用dup2系统调用,用fd覆盖掉下标为1的位置。所以上面的代码结果就是在log.txt文件里面写入"hello, Linux!"
  • 51
    点赞
  • 33
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值