万字解析文件fd,深刻理解:fd文件描述符、位图、标准输入、标准输出、标准错误、文件打开、文件关闭、Linux一切皆文件理解、进程和文件的关系、虚拟软件系统

news2025/1/17 14:10:21

建议全文阅读!!!

建议全文阅读!!!

建议全文阅读!!!

目录

文章概述

一、文件操作

1、什么叫当前路径

2、常见文件操作

(1)fopen函数

(2)fclose函数

(3)fprintf函数

3、理解文件:

4、常见系统调用

(1)open接口

[1]umask掩码

[2]位图

(2)close接口

(3)write接口

二、文件描述符fd

1、打开文件

2、标准输入、输出、错误

3、编程语言的本质


文章概述

什么是文件?
打开文件是什么意思?
怎么打开文件?
关闭文件又是什么意思?
怎么关闭文件?
打开的文件怎么管理?
关闭的文件又怎么管理?
为什么要管理?
如何管理?
文件路径是什么意思?
什么是位图?
我们不能仅从语言角度去理解文件,还需要站在操作系统的角度去理解才能全面。

所以,本文将会从操作系统的角度去深刻理解文件。
同时,回答上述的若干问题。

一、文件操作

1、什么叫当前路径

什么叫做当前路径?
当你运行一个程序的时候,你的程序就会变成一个进程
我们的路径,就是进程在执行的当前路径

查看当前路径:

pwd

2、常见文件操作

(1)fopen函数

FILE *fopen(const char *filename, const char *mode);

参数:

filename 参数是一个字符串,指定要打开的文件名或路径。
mode 参数是一个字符串,指定文件的打开模式,可以是:

"r"(只读)、

"w"(写入,如果文件存在则清空)、

"a"(追加)、

"r+"(读写,文件必须存在)、

"w+"(读写,如果文件存在则清空)、

"a+"(读写,追加到文件末尾)等。

返回值:
如果成功打开文件,则返回指向 FILE 类型结构的指针,该指针用于后续的文件操作。
如果打开失败,返回 NULL,并且通过检查 errno 变量可以确定失败的具体原因。

示例:

FILE *fp;
fp = fopen("example.txt", "r");
if (fp == NULL) {
    perror("Error opening file");
    exit(EXIT_FAILURE);
}

(2)fclose函数

int fclose(FILE *stream);

参数:

stream 是一个指向 FILE 结构的指针,指定要关闭的文件。

返回值:
如果成功关闭文件,则返回 0。
如果关闭失败,则返回 EOF(通常为 -1),并且可以通过检查 errno 变量来确定关闭失败的具体原因。

示例:

FILE *fp;
fp = fopen("example.txt", "r");
if (fp == NULL) {
    perror("Error opening file");
    exit(EXIT_FAILURE);
}

// 操作文件...

if (fclose(fp) == 0) {
    printf("File closed successfully.\n");
} else {
    perror("Error closing file");
    exit(EXIT_FAILURE);
}

(3)fprintf函数

int fprintf(FILE *stream, const char *format, ...);

参数:

stream 是一个指向 FILE 结构的指针,指定要写入的目标文件。
format 是一个格式化字符串,类似于 printf 函数中的格式化字符串,用于指定输出的格式。

返回值:

返回成功写入的字符数,如果出错则返回一个负数。


默认打开文件的时候,清空文件内容
a打开方式:append追加方式写入文件,不会清空原文件内容

3、理解文件:

(1)操作文件,本质是进程在控制文件,强调的是进程和文件的关系
(2)用户打开文件时:
找到对应磁盘->但磁盘是外部设备->把文件向磁盘写入:本质是向硬件中写入
但是用户没有权力直接写入底层硬件,因为硬件由操作系统直接管理
所以,要通过操作系统接口,系统调用
这些系统调用提供给用户进行和底层数据的控制交互

但是,我们对文件的操作都是c语言/c++等语言的函数,例如上述三个函数
我们并使用到系统调用啊?
系统调用在哪里?
然而,事实上,所有的语言函数
本质上都是对系统调用的封装
后面我们再细谈

4、常见系统调用

(1)open接口

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

两个参数的open函数,一般是打开已经存在的文件
三个参数的open函数,一般是打开没有存在的文件

参数:

pathname 是一个字符串,表示要打开的文件路径。
flags 是打开文件的标志位,例如 O_RDONLY(只读)、O_WRONLY(只写)、O_RDWR(读写)等。
mode 是文件的权限,通常与 flags 参数中的某些标志结合使用,用于指定文件的创建模式。

返回值:

如果成功,返回一个新的文件描述符,用于后续的文件操作。
如果出错,返回 -1。

[1]umask掩码

open函数的第三个参数怎么理解?

第三个参数,是设置文件创建的初始权限,例如:0666:wr_ 0777:wrx

当我们创建一个新的文件时,会有一个默认的文件权限:标红部分

初始文件权限,等于你初始设置的权限 按位与 umask掩码值

查看掩码:一般默认掩码是0022

umask

动态调整掩码:设置当前进程下的掩码,掩码使用规则是:就近原则

umask(num);
[2]位图

例如,一个整数有32位,用比特位来进行标志位的传递

什么是位图?是一种标记位传参
这些标记位都是一个一个的宏,这些宏只有一个1

例如:

1是:0000 0001

2是:0000 0010

4是:0000 0100

8是:0000 1000

...

以此类推

根据上述,我假设,

参数为1,是一个功能A

参数为2,是一个功能B

参数为3,是一个功能C

....

那么,当我同时想要A和B这两个功能时

按理来说,我就要两个参数

也就是说

要传2个参数

那么我的函数对应的就要有两个形参

那我要向同时想要ABC功能呢?
我的函数就要设计成三个相残

以此类推

不好解决

于是我怎么办呢?

传参还是一个形参int

当我想要A和B功能结合时

传参为:1 | 2

        即:0000 0001

             | 0000 0010

 结果为:0000 0011 = 3

于是,对应的函数中,我就将之0000 0011 = 3设置为A和B的功能结合

所以,我就只用一个参数,做到了两个参数的结合

这种设计方式,遇用到了比特位的运用

所以,叫做位图设计

下面我将会设计一个简单的位图程序
让你更好的理解什么是位图的设计方案

#include<stdio.h>
#define ONE 1
#define TWO ONE<<1
#define THREE TWO<<1
#define FOUR THREE<<1

void func(int flag)
{
  if(flag == ONE)
  {
    printf("NOE\n");
  }
  else if(flag == (TWO |ONE))
  {
    printf("TWO and NOE\n");
  }
  else if(flag == (THREE | TWO))
  {
   printf("THREE and TWO\n");  
  }
  else if(flag == (FOUR|THREE))
  {
   printf("FOUR and THREE\n");  
  }
  else
  {}
}
int main()
{
      func(ONE);
     func(ONE | TWO);
     func(FOUR | THREE); 

  return 0;
}

在此函数的设计思路上,去设计其他功能的函数,就是位图设计模式

那么,当有了这种设计模式后
当一个函数需要传参时,我们就不需要笨重的写多个参数
而是直接使用标记位模式,设计宏,设计代码

根据上述的理解,现在我们来理解open函数的第二个参数
这些参数就是一些宏,每个宏只有一个比特位为1
对应不同的选项
当两个不通过的选项(宏)用按位或 | 结合起来时
就会将之不同的选项功能结合起来
例如: 01 | 10 = 11
这就实现了只用一个参数,就实现了多个功能的选择组合

下列是open函数的位图宏参数列表

位图宏参数功能描述
O_RDONLY只读打开文件。
O_WRONLY只写打开文件。
O_RDWR读写打开文件。
O_CREAT如果文件不存在,则创建文件。
O_EXCLO_CREAT一同使用时,要求文件必须不存在,否则返回错误。
O_APPEND追加写入文件末尾。
O_TRUNC如果文件已存在且是普通文件,则将其长度截断为0。
O_NONBLOCK非阻塞模式打开。
O_SYNC每次write操作都要等待物理I/O完成。
O_DIRECTORY如果打开的是一个目录,则要求它是一个目录文件。
O_NOFOLLOW如果path是一个符号链接,则不会跟随符号链接。
O_CLOEXEC在进程执行exec系列函数时,自动关闭该文件描述符。

而,我们c语言的fopen以为w方式打开文件时
功能是:如果有文件,清空原文件的所有内容、没有文件就创建
你发现了什么?
你会发现,这个功能,不就是系统调用open中,O_WEONLY和O_CREAT功能的组合吗?
所以,本质上,c语言的fopen这个函数
就是对系统调用open功能函数的组合封装

运用示例:

#include <fcntl.h>
#include <unistd.h>
#include <stdio.h>

int main() {
    int fd;
    fd = open("file.txt", O_WRONLY | O_CREAT, 0644);
    if (fd == -1) {
        perror("Failed to open file");
        return 1;
    }
    close(fd);
    return 0;
}

(2)close接口

#include <unistd.h>
int close(int fd);

参数:

fd:要关闭的文件描述符。

返回值:

如果成功,返回值为0。
如果失败,返回值为-1,并设置全局变量 errno 来指示错误类型。

(3)write接口

#include <unistd.h>
ssize_t write(int fd, const void *buf, size_t count);

参数:

fd:要写入的文件描述符。
buf:指向要写入数据的缓冲区。
count:要写入的字节数。


返回值:

如果成功,返回实际写入的字节数(非负值)。
如果出错,返回值为-1,并设置 errno 来指示错误类型。

上述函数的返回值都是一个int整数,叫做文件描述符fd
0:标注你输入 键盘
1:标准输出 屏幕
2:标准错误 屏幕

二、文件描述符fd

什么是fd?是一个数组的下标!

我们知道,操作系统运行时,会有很多进程在运行
同时,也会打开很多文件
在文件没有被打开之前,文件是存在磁盘中的
那么,打开文件,是谁打开的呢?
是进程打开的
当要打开一个文件时,进程将文件的数据加载到文件内核级的缓存块中
因此,打开文件:本质是进程打开文件
而一个进程能打开很多文件
同时,系统当中可以存在许多进程
所以,在操作系统内部存在大量的文件
这些文件,可能有上万、上几十万个
有些是关闭,又是打开的
那么,操作系统如何对这些大量的文件进行管理呢?
例如:
这个文件是那个进程打开的?
以什么方式打开的?
什么时候打开?
什么时候关闭?
谁来关闭?
.....
同样的,六字真言:先描述,再组织
为了更好的管理文件
所有被打开的文件,操作系统都会对其创建一个结构体struct file,在Linux内核定义如下:

struct file {
    union {
        struct llist_node fu_llist;
        struct rcu_head fu_rcuhead;
    } f_u;
    struct path f_path;
    struct inode *f_inode;  /* 文件对应的inode */
    const struct file_operations *f_op;  /* 文件操作函数表 */
    atomic_long_t f_count;
    unsigned int f_flags;  /* 文件标志 */
    fmode_t f_mode;  /* 文件的访问权限和打开模式 */
    loff_t f_pos;  /* 文件当前的读写位置 */
    struct fown_struct f_owner;
    const struct cred *f_cred;
    struct file_ra_state f_ra;
    u64 f_version;
#ifdef CONFIG_SECURITY
    void *f_security;
#endif
    /* needed for tty driver, and maybe others */
    void *private_data;

#ifdef CONFIG_EPOLL
    /* Used by fs/eventpoll.c to link all the hooks to this file */
    struct list_head f_ep_links;
    struct list_head f_tfile_llink;
#endif /* #ifdef CONFIG_EPOLL */
    struct address_space *f_mapping;
#ifdef CONFIG_MMU
    struct vm_fault *f_vm_fault; /* fault info */
#endif
    /* Open file handle status information */
    u64 f_version;
    atomic_t f_count;
    spinlock_t f_lock;
    u32 f_flags;  /* open flags */
    fmode_t f_mode;  /* access mode */
    struct path f_path;
    struct inode *f_inode;  /* associated inode */
    const struct file_operations *f_op;  /* file operations table */
    struct device *f_owner;  /* device */
    const struct cred *f_cred;  /* credentials */
    struct file_ra_state f_ra;  /* read-ahead state */
    void *f_security;  /* security descriptor */
    void *private_data;  /* instance data */
};

我们说过,文件 = 属性 + 内容
而这个struct file结构体对象
就是用来描述一个个被打开的文件
这些被打开的文件结构体对象,就被操作系统用双向链表组织起来,成为一个双向链表
于是,操作系统对被打开文件的管理
就变成了对一个双向链表的增删查改
与此同时,在文件被打开时,
操作系统还会给对应别打开的文件,分配一个叫做文件内核级的缓存块
用于文件数据的存放,当文件被进程打开时
进程就会把文件的数据加载到该缓存块

可是,这只解决了对文件的管理
对于文件和进程之间的从属关系,如何描述组织呢?
所以,为了描述这个关系
在进程的PCB(task_struct)结构体对象中
存在一个指针变量:struct files_struct *file
这个指针变量指向一个数组:struct file* fd_array[N]
这个fd_array[N]数组内部的元素,是一个一个的file文件对象的指针
而指针,就是地址;
而地址,就是映射
所以,这些指针就指向,被该进程打开的文件
而fd_array[N]是一个数组
既然是数组,那么就必然会有下标

这个数组的下标,就对应着一个个被打开的文件的指针地址
所以,当用户想要访问一个进程下的文件
而每一个进程的PCB是唯一的,所以文件管理数组也唯一的
只要是进程打开一个文件,就会把文件的*file添加进去
我只需要返回这个进程PCB结构体下,文件管理数组的下标
是不是就可以找到对应的文件对象了?
因为只要对上提供这个下标
我就可以顺藤摸瓜,找到对文件管理数组下标对应的文件指针
即可找到文件

因此,
open函数的返回值,为int fd
这个fd到底是什么?
fd就是下标!!
即文件描述符

因此,观察下述三个系统调用函数:

 ssize_t write(int fd, const void *buf, size_t count);
 ssize_t read(int fd, void *buf, size_t count);
int close(int fd);

他们的参数,都有一个int fd
也就是说,当打开一个文件时,
要对这个文件进行写、读、操作
首先要解决对谁写?对谁读?关闭谁?
所以,清一色的,都传递open时返回的文件描述符:fd参数

int open(const char *pathname, int flags);

让write、read、close这三个函数
用fd去找到对应的文件
然后执行相关操作

同时,对文件的读写等操作的数据改动
都是在文件缓存块内进行

然后由操作系统刷新到磁盘对应位置
所以,读取文件,本质上就是从文件的文件内核缓存块内加载数据到应用层
而写文件,以及是在文件缓存块进行
所以,无论是读文件还是写文件
都是内存级别的操作
然后再由操作系统在合适的时机刷新到磁盘中
所以,本质来说,read、write都是拷贝函数,就是般数据的函数
这里要注意:
磁盘中的文件数据,必须要先加载到文件缓存块中(内存)
才可以进行修改

1、打开文件

所以,当open一个文件时,到底在干什么?
1、创建file文件结构体
2、开辟文件缓存区的空间,加载文件数据(从磁盘中加载,有可能延后)
3、查看进程的文件描述符表
4、将file文件地址,填入对应的表下标中
5、返回下标,即fd

2、标准输入、输出、错误

当我们理解了上述的陈述之后
我们再接着来理解这三个东西:
0:标注输入 键盘
1:标准输出 显示器
2:标准错误 显示器

当我们打开在一个进程下只有一个文件时,我们查看该文件的fd,按理来说,一个进程下只有一个我们创建的文件,因此,进程的文件描述符号表对应的,只有一个文件,数组中文件的下标应为0,所以文件的fd应该是0,但是实际上却不是,而是3

所以,为什么是3呢?也就是说前面还有0、1、2三个文件,这三个文件是什么?

下面解释:
鼠标、键盘、显示器,都有对应的文件描述符
既然有文件描述符
说明有对应的file结构体
可是,这三个货明明是硬件
为什么就变成了文件呢?
怎么做到的呢?
我又该怎么理解呢?
例如说,是文件,就说明有读写
读键盘,我能理解
但是写键盘,是什么意思?
往键盘里写数据?
啥?
对不起,我理解不了一点

所有想要理解上述的问题,需要进一步理解Linux一切皆文件的理念
我们理解这个理念的困难,在于如何理解硬件
怎么把硬件也视为文件?

每一个硬件设备,我们关注的主要是两点:属性和操作方法
首先是属性
虽然每一个硬件的属性都不同
但是,我们可以抽象属性类别
创建硬件设备的结构体device
结构体内部有着诸如:
设备名、设备状态、厂商等属性
于是,每一个设备都对应着一个device结构对对象
于是所有的硬件设备都是属于同一个结构体类型
仅对象本身数据的不同
于是,我们就通过先描述,再组织
将硬件设备管理起来了
上述是属性

接下来是操作方法
每一个设备都有着对应的操作方法
例如,读操作和写操作
现在我们来理解键盘
键盘有读操作,这个好理解
但是,没有写操作
那怎么办呢?
我们将键盘的写操作设置为空方法

对应着的,
显示器写操作
但是没有读操作
所以,我们把显示器的读操作设置为空

所以,每一个设别都有着不同的操作方法
而每个硬件设备的操作方法,例如读写操作
在出厂商设置之前,都已经设计好了对应的接口方法
这体现在驱动层

而在Linux操作系统中
每一个驱动设备,都创建一个对应的struct file结构体
在这个结构体内部
有属性,这些属性就是设备的属性数据
还有函数指针,
所以,尽管每一个设备的操作方法不一样
但是,我可以把方法的返回值、参数设置成一样的
然后让这些函数指针指向底层的硬件设备的操作方法
所以,这个结构体还有一张操作底层指针表,指向底层的操作方法

于是,经过上述的组织之后
我要用键盘的read方法,则指向底层键盘的read方法
我要用鼠标的read方法,则指向底层鼠标的read方法
我要用屏幕的read方法,则指向底层鼠标的write方法
以此类推...
上述实现这种组织的技术
其实就是多态! 
我们再回头来看struct file这个结构体
有属性、有方法
但是c语言的结构体内不能写函数
所以我们使用函数指针
所以,这个struct file结构体,其实就是一个类
所以,从一切皆对象的理解角度来看
文件也是对象的一种

于是,站在struct file这个结构体对象之上
我们就不用再关心底层的硬件的方法到底是怎样的
当我们想调用一个硬件设备的方法时
例如要读取键盘
就是找到键盘的结构体对象
然后直接调用其中的read方法即可

于是,通过上述的组织方式
现在站在Linxu的角度来看
这些硬件,也视为文件
所以,一切皆文件!

所以对于硬件这一层,在Linux下叫做vfs,即vitural file system,虚拟软件系统

所以,当我们底层的硬件性质不一样时,我们可以在其上设计一层软件
屏蔽掉底层硬件上的差异
这样就可以以统一的视角去看待、组织、管理底层

3、编程语言的本质

在系统内,访问文件时,只认文件描述符
可是,c语言中的fopen、fclose函数等
返回值是一个文件类型指针:FILE*
哪里有文件描述符?
这个FILE是什么?
是一个c语言提供的结构体类型
而这个FILE结构体内部
封装了文件描述符!!!

所以,本质上,c语言的所有文件操作函数,底层都是对系统调用的封装


所以,我们写代码可以使用系统调用,也可以使用语言提供的方法
可以,既然已经有了系统调用,为什么还要有一个语言呢?
直接用系统调用函数不就好了吗?
c语言为什么这么做?
这是因为,系统不同,系统调用接口不同
例如,你在Linux下的系统调用,在mac、windows下就可能不一样
在Linux下,你是一个参数;mac下是2个参数;windows下是3个参数
根本没法兼容
于是,你使用Linux的系统调用写出来的代码
只能在Linux下编译运行
但是,不能拿到mac、windows下运行
也就是说,代码不具备跨平台性

所以,c语言/c++/java等语言,解决了上述的问题
在c语言本身的标准库中
对每一个操作系统的环境的系统调用都进行了封装
再用条件编译进行调整兼容
就这样,你写的同一份代码
就可以在不同的系统环境中编译运行

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

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

相关文章

YASKAWA安川直驱电机DD马达Σ-7系列介绍

随着智能制造的兴起&#xff0c;对设备精度、效率及可靠性的要求日益提升&#xff0c;安川Σ-7系列无需减速机即可直接驱动负载的“直驱伺服电机”&#xff0c;以其独特的优势正逐步成为众多高精度、高效率应用场景下的运动控制解决方案。 一、直驱技术的革命性突破 传统伺服…

When can a sum and integral be interchanged?

https://math.stackexchange.com/questions/83721/when-can-a-sum-and-integral-be-interchanged https://math.stackexchange.com/questions/1334907/reversing-the-order-of-integration-and-summation

【大模型从入门到精通12】openAI API 提示链的力量3

这里写目录标题 实践问题 实践部分场景概述场景步骤初始产品询问故障排除请求保修问题额外产品推荐 示例实现 实践问题 编写一个名为retrieve_model_response的Python函数&#xff0c;该函数接受一个消息序列作为输入&#xff0c;并根据给定参数返回模型的响应。包括模型、温度…

如何去掉el-input自带边框

<style lang"scss" scoped>::v-deep .inputDeep .el-input__inner {border: none !important;box-shadow: none !important;padding: 0px; }</style> //先定义一个类名 <el-input v-model"form.name" class"inputDeep"><…

7.3.1.算法设计与分析-总结及真题讲解

总结 分治法特征&#xff1a;把一个问题拆分成多个小规模的相同子问题&#xff0c;一般可用递归解决。 经典问题&#xff1a;斐波那契数列、归并排序、快速排序、矩阵乘法、二分搜索、大整数乘法、汉诺塔 回溯法特征&#xff1a;系统的搜索一个问题的所有解或任一解。 经典问题…

10个理由告诉你,为什么鸿蒙是下一个职业风口!

在当今科技飞速发展的时代&#xff0c;新的技术和趋势不断涌现&#xff0c;为人们带来了前所未有的机遇和挑战。鸿蒙操作系统作为我国自主研发的创新成果&#xff0c;正逐渐成为科技领域的焦点&#xff0c;被认为是下一个职业风口。 10个理由告诉你&#xff0c;为什么鸿蒙是下一…

C++(week15): C++提高:(五)Redis数据库

文章目录 零、Redis的安装、API1.redis、hiredis、redis-plus-plus安装2.HiRedis的API 一、Redis数据库的基本概念1.关系型数据库与非关系型数据库的区别2.非关系型数据库的分离3.Redis的概念4.Redis的特性5.Redis的优点 二、Redis常用命令三、Redis的五大数据类型及其命令1.st…

清除 Nuxt 状态缓存:clearNuxtState

title: 清除 Nuxt 状态缓存&#xff1a;clearNuxtState date: 2024/8/7 updated: 2024/8/7 author: cmdragon excerpt: 摘要&#xff1a;本文介绍了Nuxt.js框架中clearNuxtState方法的使用&#xff0c;该方法用于清除useState管理的状态缓存&#xff0c;确保应用状态的有效性…

Apache POI 实现 Excel 表格下载

这里以苍穹外卖中数据导出功能为例&#xff0c;记录下 Apache POI 导出 Excel 表格的过程。 首先在 pom.xml 中导入相关依赖 <!-- poi 用于操作 excel 表格--> <dependency><groupId>org.apache.poi</groupId><artifactId>poi</artifactId&…

详细LVS实验配置

一、LVS的NAT模式 1、实验环境 主机名ipVIP角色lvs192.168.0.100172.25.254.100调度器webserver1192.168.0.10&#xff0c;网关192.168.0.100null 真实服务器&#xff08; RS &#xff09; webserver2192.168.0.20&#xff0c;网关192.168.0.100null 真实服务器&#xff08; R…

【C语言初阶】C语言操作符全攻略:提升编程效率的关键步骤

&#x1f4dd;个人主页&#x1f339;&#xff1a;Eternity._ ⏩收录专栏⏪&#xff1a;C语言 “ 登神长阶 ” &#x1f921;往期回顾&#x1f921;&#xff1a;C语言数组 &#x1f339;&#x1f339;期待您的关注 &#x1f339;&#x1f339; ❀操作符 &#x1f4d2;1. 算术操作…

从0开始搭建vue + flask 旅游景点数据分析系统( 八):美化前端可视化图形

这一期来美化我们仅有的三个可视化图形&#xff08;可怜&#xff09;&#xff0c;毕竟&#xff0c;帅是一辈子的事。 1 折线图改面积图&#xff08;渐变色&#xff09; 需求&#xff1a;折线图改为蓝色的面积图&#xff0c;并且有一点的渐变色。 这个非常简单&#xff0c;只…

提升效率神器!2024年Windows平台录屏工具

现在生活中经常会用到录屏工具&#xff0c;比如会议记录、比如课程教学、比如游戏瞬间等等。如何选择一款适合WIN10录屏的工具就值得我们研究一下。 1.福昕REC大师 链接直达&#xff1a;https://www.foxitsoftware.cn/REC/ 这款软件的界面设计极其简约直观&#xff0c;一眼…

代码随想录第二十三天|动态规划(7)

目录 LeetCode 188. 买卖股票的最佳时机 IV LeetCode 309. 买卖股票的最佳时机含冷冻期 LeetCode 714. 买卖股票的最佳时机含手续费。 总结 LeetCode 188. 买卖股票的最佳时机 IV 题目链接&#xff1a;LeetCode 188. 买卖股票的最佳时机 IV 思想&#xff1a;本题跟之前的…

Kevin De Rug VoxEdit 竞赛来了!

让 Kevin de Rug 再现&#xff0c;去世界的某个地方冒险。 让你的创造力更上一层楼。以 Pixelmon 的 Kevin de Rug 为主题&#xff0c;在这场精彩的 VoxEdit 竞赛中释放你惊人的体素技能。 主题&#xff1a;让 Kevin de Rug 再现&#xff0c;去世界的某个地方冒险。 Kevin 被…

Linux系统驱动(十一)GPIO子系统

文章目录 一、GPIO子系统&#xff08;一&#xff09;框架结构&#xff08;二&#xff09;GPIO子系统的API&#xff08;三&#xff09;gpio子系统控制LED灯的设备树1. 画出硬件连接图2. 找出控制器的设备树3. 参考内核帮助文档 二、使用GPIO子系统实现流水灯1. 设备树文件&#…

Win11解压文件Cpu占用率过高?解决方法在此!

在Win11电脑操作中&#xff0c;用户遇到解压文件时CPU占有率过高的问题&#xff0c;不知道要如何操作才能解决该问题&#xff1f;接下来系统之家小编给大家分享几种不同的解决方法&#xff0c;帮助大家轻松解决问题&#xff0c;降低Win11电脑CPU占有率&#xff0c;提升Win11电脑…

记一次框架升级

背景 随着公司业务的不断扩展&#xff0c;新技术的更新换代&#xff0c;企业内部免不了会对软硬件进行升级&#xff0c;淘汰老旧的组件和实现方案&#xff0c;更新一波技术栈。这不&#xff0c;最近我们公司就面临这么一个难题&#xff1a;旧版本的组件上发现漏洞&#xff0c;为…

Fiddler代理后浏览器无法上网啥情况

当使用Fiddler作为代理服务器后&#xff0c;浏览器无法上网的情况通常是由以下几个原因造成的&#xff1a; 代理服务器配置不正确&#xff1a; 确保在浏览器或其他客户端中正确配置了Fiddler作为代理服务器。代理服务器地址应为运行Fiddler的计算机的局域网IP地址&#xff0c;端…

【Canvas与艺术】黄色立体感放射光芒五角星

【成图】 【代码】 <!DOCTYPE html> <html lang"utf-8"> <meta http-equiv"Content-Type" content"text/html; charsetutf-8"/> <head><title>黄色立体感放射光芒五角星</title><style type"text/c…