【Linux】基础IO——系统文件IOfd重定向理解

news2024/11/16 1:26:51

文章目录

    • 一、回顾C文件接口
      • 1.打开和关闭
      • 2.读写文件
      • 3.细节
    • 二、系统文件I/O
      • 1.open和close
        • umask小细节
      • 2.read和write
        • 1.write
        • 2.read
        • 3.小总结
    • 三、理解文件
    • 四、文件描述符fd
        • 1.引入
        • 2.理解
        • 3.分配规则
        • 4.close(1)问题
    • 五、重定向
        • 1.重定向
        • 2.接口
        • 3.追加重定向
        • 4.输入重定向
    • 六、Linux一切皆文件

一、回顾C文件接口

1.打开和关闭

对于C语言的文件操作,首先我们需要打开(fopen)文件,打开失败将会返回NULL ,而打开成功则返回文件的指针(FILE*)

最后我们则需要关闭(fclose)文件。

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

filename为文件名(包括文件路径),mode为打开方式,它们都是字符串

int fclose(FILE *stream)

stream – 这是指向 FILE 对象的指针,该 FILE 对象指定了要被关闭的流。

下面我们用C语言来实现打开和关闭文件:

image-20221214231013245

2.读写文件

我们可以通过fgets和fputs以字符串形式进行读写,也可以通过fprint和fscanf进行格式化读写

int fputs (const char * str, FILE * stream );
char * fgets (char * str, int num, FILE * stream );
int fprintf (FILE * stream, const char * format, ... );
int fscanf (FILE * stream, const char * format, ... );

在这个地方,我们先重新理解下当前路径:

当前路径:一个进程运行起来的时候,每个进程都会去记录自己当前所处的工作路径

所有当前路径也就是当前进程的工作路径,可以被修改,所以每个进程都有自己的当前路径

image-20221214225056132

文件操作的本质是进程和被打开文件的关系

文件操作中r和w分别代表读和写,r+(读写)代表不存在则出错,w+(读写)代表不存在则创建,a(append)进行追加,追加也是写入,a+()也是读写,写是追加。

fprintf+w:

image-20221214231527583

fgets+r

fgets会给字符串结尾添加\0

image-20221214232546954

运行的结果和文件的内容每行都多出了一行,这是因为在读取的时候按行打印,把\n多读了,所以我们可以处理一下\n:

image-20221214233037822

a追加

image-20221214235425674

对于C语言文件操作接口我们就先说到这里。这些都是我们之前所学的,在这里简单复习一下。

3.细节

对于C语言文件操作存在一些细节在这个地方提出来:

1.以w方式单纯的打开文件,c会自动清空内部的数据

image-20221215084018711

为什么会自动清空内部的数据,这是一个值得思考的问题:这是因为O_TRUNC(这文件内容做清空),不懂的话那就先继续往下看把。

2.使用比特位传递选项

在C语言中,我们传标记位一个整数标记位一般传一个整数,而一个整数有32个比特位,所以我们可以通过比特位来传递选项。下面,我们要说的就是使用比特位来传递选项,一个比特位一个选项,比特位位置不能重复:

image-20221216092931982

image-20221216095822185

通过|传递参数,这样就能传递多个标志位了。


二、系统文件I/O

文件操作除了上面我们所说的语言级别的文件操作,还有系统级别的文件操作调用接口。

我们知道:普通文件默认创建的是664:一个文件形成的时候有默认文件的野码,普通文件创建的时候默认的起始权限是666,在形成文件的时候666&~umask。

1.open和close

fopen调用底层实际上是open:

  • open

我们可以来看看open的参数以及返回值(man 2 open):

image-20221216105500024

//头文件
#include <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);//文件不存在
//pathname:打开文件名
//flags:标志位。O_RDONLY:只读 O_WRONLY:只写 O_RDWR:读写
//返回值
return the new file descriptor, or -1 if an error occurred (in which case, errno is set appropriately)
//成功:打开文件描述符
//失败:返回-1
  • close

man 2 close

//头文件
#include <unistd.h>
//参数
int close(int fd);

同时,O_CREAT:文件不存在,则需要我们去创建它,并不会自动创建好哈。也要使用mode选项,来指明此时新文件的权限。

注意:O_CREAT是一个建议选项,文件存在还是不存在都可以使用

image-20221216110218005

1.我们发现了open并没有去帮我们自动创建文件。以写的方式去创建方式并没有自动创建,想啥呢哈哈。而在C语言封装了会帮我们自动创建,但是对于系统接口我们需要加上O_CREAT(文件不存在自动创建).最终成功帮我们自动创建成功!!!

**2.但是对于log.txt文件创建了,权限是乱的,但是文件默认以什么权限创建?我们默认情况下目录以777,普通文件以666开始,**这些都是通过open的第三个参数mode选项设置权限的,设定创建默认文件的权限

使用mode选项设置权限,我们一起来看一看:

image-20221216110343536

我们可以设置unmask改变权限:

image-20221216115723673

此时log.txt的权限才是664,与C语言创建的默认权限就一致了!

umask小细节

我们如果想创建文件的权限如果不想受系统影响,也可以自己定义创建文件的野码

换句话来说:我们默认使用的是系统给我们所提供的umak,也就是父shell给我们提供的,而子进程会继承,所以我们很容易知道子进程对应的野码是多少。但是如果我们不想受系统野码影响限制权限,我们就可以在我们自己的子进程使用umask(0)清空野码(也就是上面的代码)image-20221218234930208

但是此时父进程shell的umask结果还是0002,我们改变的是子进程的文件权限,因为进程具有独立性,并不会影响父进程的umask

这就是在这里所说的umask小细节,值得注意一下。

2.read和write

1.write

文件打开和关闭说完之后,自然就是我们的写入接口了

//man 2 write
#include <unistd.h>
//把一个文件先描述符输入
ssize_t write(int fd, const void *buf, size_t count);

fd:所写的文件

buf:缓冲区数据,参数是void*,这从侧面说明了很多的东西:我们之前所说,文件读取分为文本类和二进制类,这是对于语言所提供的文件读取的分类。但是在操作系统看来,都是void*,也就是二进制!

count:缓冲区所写的字节个数

返回值:返回写入的字节数,在这里我们并不太需要关注返回值

sprintf:将特定的内容格式化到字符串

话不多说,我们直接代码练习:

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <assert.h>
#define FILE_NAME "log.txt"
int main()
{
    umask(0);
    int fd = open(FILE_NAME,O_WRONLY | O_CREAT,0666);
    if(fd<0)
    {
        perror("open");
        return 1;
    }
    int cnt = 5;
    char outBuffer[64];
    while(cnt)
    {
        sprintf(outBuffer,"%s:%d\n","helloworld",cnt--);
        
        write(fd,outBuffer,strlen(outBuffer)+1);  //注意这个地方+1了                                                 
    }
    close(fd);
}

image-20221219001548321

结果很正常,符合我们的预期,但是我们以文本打开:

  • string+1出现乱码问题

image-20221219001235543

出现上面这种情况乱码问题,实际是当我们向文件写入string的时候,要不要加1的问题?

\0作为字符串的结尾,是C语言的规定,和文件并没有什么关系,是有效内容结尾。所以我们并不需要+1。这又是一个小细节

image-20221219003844530

  • 清空问题

image-20221219004148371

我们发现,在这里,当我们重新打印内容时,居然还残留着上一次所打印的helloworld,这并没有帮我们自动清空内容,需要我们自己添加选项内容(而在C语言中,我们一开始说的细节:以w方式单纯的打开文件,c会自动清空内部的数据,这是封装好的)。

清空内容需要带上O_TRUNC:

image-20221219005441008

此时如果我们就打开和关闭,所有内容也会被清空:

#include <stdio.h>
#include <unistd.h>
#include <string.h>
#include <sys/types.h>
#include <sys/stat.h>
#include <fcntl.h>
#include <assert.h>
#define FILE_NAME "log.txt" 
int main()
{
    umask(0);
    int fd = open(FILE_NAME,O_WRONLY | O_CREAT | O_TRUNC,0666);
    if(fd<0)
    {
        perror("open");
        return 1;
    }                                                                                             
    close(fd);
}

image-20221219005845857

通过上面的这些内容,我们知道,在操作系统上层一个简单的"w"选项,操作系统底层就需要我们传O_WROLY(写入),O_CREAT(不存在则创建),O_TRUNC(清空),以及传入属性!这就是C语言与系统接口的联系。

  • 追加O_APPEND

image-20221219091326874

2.read

image-20221219212743663

从一个文件描述符中读取文件

//头文件
#include <unistd.h>
//返回值ssize_t系统定制类型
ssize_t read(int fd, void *buf, size_t count);

此时读文件需要用到选项O_RDONLY

返回值:

image-20221219212857501

成功返回读取到多少个字节,0代表读到文件结尾。

读文件的前提是文件已经是存在的了,不涉及创建和权限的问题。

下面,进入代码演示环节:

image-20221219213456369

3.小总结

小总结:我们上面学习了open/close/write/read接口,当然还有lseek接口,这里就不展开说了

实际上,这上面四个系统调用接口就对应着C语言的fopen/fclose/fwrite/fread,以及fseek库接口。

image-20221219214336022

上面的 fopen fclose fread fwrite 都是C标准库当中的函数,我们称之为库函数(libc)。
而open close read write lseek 都属于系统提供的接口,称之为系统调用接口


三、理解文件

1.文件操作的本质:进程和被打开文件的关系

2.进程可以打开多个文件,这也就意味着系统中一定会存在大量的被打开的文件,然而被打开的文件则需要被操作系统管理,我们知道,管理的本质就是先描述在组织,所以操作系统为了管理对应的打开文件,操作系统必定要为文件创建对应的内核数据结构来标识文件,这个内核数据结构就是struct file{}结构体(与C语言的FILE没有关系哦);包含了文件的大部分属性

**3.而进程和被打开的文件如何关联,也就是说进程和被打开文件的关系是如何维护的?**通过文件打开(open)的返回值和文件描述符进行联系。

下面我们通过代码来看一看返回值究竟是多少

#include <stdio.h>  
#include <unistd.h>  
#include <string.h>  
#include <sys/types.h>  
#include <sys/stat.h>  
#include <fcntl.h>  
#include <assert.h>  
#define FILE_NAME(number) "log.txt"#number     
int main()  
{
    umask(0);  
    int fd0 =open(FILE_NAME(1),O_WRONLY | O_CREAT | O_APPEND,0666);  
    int fd1 =open(FILE_NAME(2),O_WRONLY | O_CREAT | O_APPEND,0666);  
    int fd2 =open(FILE_NAME(3),O_WRONLY | O_CREAT | O_APPEND,0666);  
    int fd3 =open(FILE_NAME(4),O_WRONLY | O_CREAT | O_APPEND,0666);  、
    int fd4 =open(FILE_NAME(5),O_WRONLY | O_CREAT | O_APPEND,0666);  
    printf("fd: %d\n",fd0);  
    printf("fd: %d\n",fd1);  
    printf("fd: %d\n",fd2);  
    printf("fd: %d\n",fd3);  
    printf("fd: %d\n",fd4);  
    close(fd0);
    close(fd1);
    close(fd2);
    close(fd3);
    close(fd4);    
}

image-20221219215843210

为什么从3开始???👇


四、文件描述符fd

1.引入

看到上面的结果,open的返回值为什么是从3开始的,那0,1,2跑哪里去了呢,而且还是连续的小整数(说到连续,我们想到的是数组下标连续)

在C语言阶段,我们知道C程序会默认打开三个标准输入输出流:stdin(标准输入设备键盘).stout(输出设备显示器).stderr(显示器)

而对于C语言的FILE,我们对其并不太了解:c语言的FILE究竟是何方神圣?这实际上是一个结构体!访问文件时,底层open必须采用系统调用,而系统调用接口访问文件必须用文件描述符,而在C语言用的并不是文件描述符,而是FILE,所以这个FILE结构体必定有一个文件描述符的字段。所以C语言不仅在接口上有封装,连数据类型都有封装。*

所以,我们可以查看到stdin,stout,stderr里面对应的值是多少:

image-20221219223133032

这就很好的解答了为什么open的返回值是从3开始的问题!因为0,1,2默认被占用,我们的C语言封装了接口,同时也封装了操作系统内的文件描述符。

此外,数字为什么从0,1,2连续的整数,文件描述符的本质是什么?

2.理解

文件描述符的本质是数组的下标

image-20221220162335757

一个文件如果没有被打开那就是在磁盘上的,而要操作文件,就需要打开文件,把文件相关的属性信息从磁盘加载到内存,操作系统中会存在大量的进程,一个进程可以打开多个文件,所以操作系统要把很多的文件在内存中管理起来,如何管理?先描述,在组织。OS为了管理每一个打开的文件,构造了struct file对象,那打开那么的文件,OS为了让进程和文件之间产生关联,进程创建struct file_struct的结构,同时里面包含了数组struct file*fd_array[]指针数组,把描述文件的结构体对象地址填充到对应的下标之中

这也就很好结社了为什么打开文件返回值为3,打开文件内核会描述struct file结构,把对应的地址填充到struct file*fd_array[]数组中的下标中去,又因为0,1,2,默认会被占用,于是从3号下标开始,对应的数组下标返回给用户,这样就能找到进程的文件描述符表,找到对应的文件了。

这也就是为什么文件操作系统读到的数是整数,而且是连续的,因为文件操作系统内标记进程和文件之间的关系就是文件描述符表,用数组标定文件内容!通过文件描述符来访问文件!

3.分配规则

文件描述符说白了就是数组的下标。下面,我们进入的是文件描述符的分配规则。

既然默认会打开0,1,2,那我们如果将其关闭呢

image-20221220180049971

一个文件被打开是进程被打开,进程的task_struct,被打开的文件struct_file,进程和被打开的文件通过文件描述符表struct files_struct里面包含一个数组struct file fd_array[]指向对应文件的数组,里面写着被打开文件的地址,下标对应着填充的文件对象。进程找到自己的文件描述符表传入对应的下标值访问对应的文件。当我们把0关掉了,没有被占用,此时如果在创建一个文件对象,会在自己的文件描述符表从小到大按照顺序寻找最小的且没有被占用的fd.*

fd的分配规则总结一句话:从小到大按照顺序寻找最小的且没有被占用的fd。而默认会把0,1,2占用了,所以一开始文件描述符是3.当我们把0关闭,0没有被占用,那文件描述符那就是0了。

看到这里,如果细心的话就会发现,前面我们都没有close(1),那close(1)会发生什么问题?👇这又引出了另外一个问题

4.close(1)问题

根据前面所说的分配规则:我们可以知道,当我们关闭1时,此时1不在指向标准输出(显示器),不在向显示器打印,当我们打开文件的时候,系统会存在文件对象,然后在把文件的地址在files_struct找一个最小的没有被使用的文件描述符,此时是1,此时就把文件的地址填入1的下标里,在把1号文件描述符返回给上层,此时fd就拿到返回值1。

但是结果是1吗?

image-20221224093013256

但是我们运行并没有看到结果,原因是因为:printf实际就是向stdout打印的。

我们向标准输出显示器进行打印的时候,默认是向stdout打印,而stdout的文件描述符封装了1,当我们进行文件操作的时候,都是在向stdout打印,而stdout的fd都是1,当1号文件描述符没变,里面的指向发生了变化,指向了新打开的文件。
注意:此时打印的结果并没有在新打开的文件里,这是因为缓冲区的问题,没有被显示出来

image-20221224093111028

此时我们刷新缓冲区,即可看到结果:

image-20221224093405188

本来我们应该把打印往显示器文件里打印,最后经过我们的一系列操作把输出的结果写到了文件里。也就是本来应该写到显示器,却写到了文件,这种特性我们称之为重定向!


五、重定向

1.重定向

重定向我们最早接触就是>输出;>>追加;<输入

重定向最典型的特征就是在上层调用不变的情况下,改变底层的数组方向:比如调用fwrite(stdout,…);无论如何调用,上层都会用到stdin(标准输入),stdout(标准输出),stderr(标准错误),也就是0,1,2不变,当我们把3号描述符的指针指向1号描述符,1本来是指向标准输出的,此时1却指向了新打开的文件.

重定向的本质就是上层的fd不变,在内核中更改fd对应的struct_file*的地址

下面,我们要介绍的就是重定向接口。

2.接口

image-20221224151325824

dup2的作用是两个文件描述符之间进行拷贝,是把fd里面的内容拷贝,这点是我们需要注意的。

dup2的参数我们需要去格外关注一下:dup2一旦重定向之后,最终剩下的都是oldfd:

image-20221224230500743

下面,我们直接通过dup2重定向把原来显示到显示器的内容却显示到文件里面:

image-20221224231252163

3.追加重定向

无非就是在打开文件的时候不需要清空直接改成追加方式O_APPEND:

image-20221224232404401

4.输入重定向

输入重定向前提条件是文件必须存在。stdin—>0,dup2(fd,0);//输入重定向

image-20221224233831347


六、Linux一切皆文件

Linux一切皆文件:

在冯诺依曼体系中,我们知道硬件有键盘、显示器、磁盘、网卡等外设,在IO过程中,外设任何的数据处理都需要把数据读到内存,处理完毕之后将内存中的数据刷新到外设当中。因为软硬件资源多,所以操作系统需要对其先描述,在组织。所以这些外设都有对应的结构体,对应着属性信息,同时,对应着自己的IO函数,具体硬件的读写方法都在应用匹配的驱动程序里。每种硬件的访问方法都是不一样的,而Linux一切皆文件是这样体现的:任何一个被打开的文件结构体对象struct file{ //各种文件的属性 }对象,不同的文件对应的读写方法不一样,struct file对象里面可以有很多的(*readp)()、(*writep)()函数指针,通过函数指针指向具体的读写方法。

站在struct file上层看来,所有的设备和文件,统一都是struct file->,就可以调用具体的设备方法了,所以在用户级看到的就是Linux下一切皆文件!

**上层调用不同的文件,底层可以调用不同的方法,在上层看来,只需要使用对应统一的文件,使用struct file,访问不同的文件,这是C语言实现多态的特征。这里struct file称为在操作系统层面上虚拟出来的文件对象vfs(虚拟文件系统)**不用关心底层差别,统一使用文件的接口方式进行文件操作

下面我们可以查看源码,看看一些细节:

PCB指向所有被打开的文件:

image-20221225092708540

image-20221225142519130

image-20221225150013937

我们所谓的关闭文件只是在表明用户给OS说已经不需要使用了,由OS决定,OS把引用计数减到0时,才被OS真正删除掉。

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

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

相关文章

信息技术 定义内涵

工作流运行 定义内涵 工作流运行是工作流模板的依次执行&#xff0c;在工作流运行时&#xff0c;用户可以随时取消或查看正在 运行的任务。由于工作流运行的模板的不同&#xff0c;运行过程中可能会产生不同的新资源&#xff0c;如数据 处理类型的工作流会产生新的数据集&…

Java框架精品项目【用于个人学习】

难度系数说明&#xff1a; 难度系数用来说明项目本身进行分析设计的难度 难度系数大于1的项目可用作参赛作品、大作业、计算机毕业设计等需求 前言 大家好&#xff0c;我是二哈喇子&#xff0c;此博文整理了各种项目需求&#xff0c;用于博主自己学习&#xff0c;当做个人笔记…

黑烟车识别抓拍系统 python

黑烟车识别抓拍系统利用Python基于YOLOv5通过道路已有卡口相机对现场画面中包括黑烟车、车牌信息、车头车尾照片、林格曼黑度等级数据回传给后台。Python是一种由Guido van Rossum开发的通用编程语言&#xff0c;它很快就变得非常流行&#xff0c;主要是因为它的简单性和代码可…

IMX6ULL学习笔记(14)——GPIO接口使用【C语言方式】

一、GPIO简介 i.MX6ULL 芯片的 GPIO 被分成 5 组,并且每组 GPIO 的数量不尽相同&#xff0c;例如 GPIO1 拥有 32 个引脚&#xff0c; GPIO2 拥有 22 个引脚&#xff0c; 其他 GPIO 分组的数量以及每个 GPIO 的功能请参考 《i.MX 6UltraLite Applications Processor Reference M…

【魔法圣诞树】代码实现详解 --多种实战编程技巧倾情打造

一、前言 本文会基于C# GDI技术 从零到一 实现一颗 魔法圣诞树&#xff01;源码和素材在文末全部都有&#xff01; 二、魔法圣诞树 对于用代码画圣诞树&#xff0c;网上各种编程语言像python、css、java、c/c我们都有见到过了&#xff0c;那么在绘图方面&#xff0c;还有一位…

从刘润的商业简史,预测互联网与能源的未来,辉煌的人生需要顺势而为

所有理所当然的现在&#xff0c;都是曾经看起来不可能的未来。 所有现在看起来不可想象的未来&#xff0c;可能都是明天理所当然的现在。 未来已来&#xff0c;只是尚未流行。 “一切历史都是当代史。”学习历史的目的&#xff0c;正是为了从中总结规律&#xff0c;然后用这些…

第四章:数据库安全性

一、数据库安全概述和控制 1、【单选题】TCSEC/TDI安全级别划分中&#xff0c;C1级需要实现的安全策略为&#xff1a; 我的答案&#xff1a;A 2、【单选题】能够对系统的数据加以标记&#xff0c;对标记的主体和客体实施强制存取控制&#xff08;MAC&#xff09;、审计等安全机…

绿盟SecXOps安全智能分析技术白皮书 工作流运行

工作流运行 定义内涵 工作流运行是工作流模板的依次执行&#xff0c;在工作流运行时&#xff0c;用户可以随时取消或查看正在 运行的任务。由于工作流运行的模板的不同&#xff0c;运行过程中可能会产生不同的新资源&#xff0c;如数据 处理类型的工作流会产生新的数据集&…

【关于时间序列的ML】项目 10 :用机器学习预测降雨

&#x1f50e;大家好&#xff0c;我是Sonhhxg_柒&#xff0c;希望你看完之后&#xff0c;能对你有所帮助&#xff0c;不足请指正&#xff01;共同学习交流&#x1f50e; &#x1f4dd;个人主页&#xff0d;Sonhhxg_柒的博客_CSDN博客 &#x1f4c3; &#x1f381;欢迎各位→点赞…

浅谈会话技术:Cookie,Session、Token

◼️ 什么是会话 会话&#xff1a; 数据交互的过程&#xff0c;在web中指 浏览器从发出一个请求到浏览器关闭&#xff0c;这个过程就是一个会话。在这个过程中&#xff0c;需要有很多的状态和数据需要我们关注&#xff0c;记录&#xff0c;这个就是我们要研究的会话 ◼️ 什么…

基于主从博弈的社区综合能源系统分布式协同优化运行策略(Matlab代码实现)

&#x1f468;‍&#x1f393;个人主页&#xff1a;研学社的博客 &#x1f4a5;&#x1f4a5;&#x1f49e;&#x1f49e;欢迎来到本博客❤️❤️&#x1f4a5;&#x1f4a5; &#x1f3c6;博主优势&#xff1a;&#x1f31e;&#x1f31e;&#x1f31e;博客内容尽量做到思维缜…

源码系列 之 HashMap

简介 HashMap可能是Java程序员最常用的集合之一了&#xff0c;面试也是常考题之一。那么我们平时了解到的特性都是怎么来的呢&#xff0c;什么一会儿是链表&#xff0c;一会儿又是红黑树。八股文选手现在已经像高中背《滕王阁序》一样&#xff0c;肌肉反应似的在脑子背过一遍了…

Github一夜爆火,不愧是阿里、字节高级架构师联合创作的Netty核心原理手册

讲到这里&#xff0c;你可能要问了&#xff1a;如果我的工作中涉及网络编程的内容并不多&#xff0c;那我是否还有必要花精力学习 Netty 呢&#xff1f;其实在互联网大厂&#xff08;阿里、腾讯、美团等&#xff09;的中高级 Java 开发面试中&#xff0c;经常会问到涉及到 Nett…

【Javassist】快速入门系列06 当检测到构造方法调用时替换方法调用的内容

系列文章目录 01 在方法体的开头或结尾插入代码 02 使用Javassist实现方法执行时间统计 03 使用Javassist实现方法异常处理 04 使用Javassist更改整个方法体 05 当有指定方法调用时替换方法调用的内容 06 当有构造方法调用时替换方法调用的内容 文章目录系列文章目录前言引入J…

python数据分析 股票预测

目录 ​​​​​​​一&#xff1a;股票预测数据分析 二&#xff1a;K线图数据分析 ​​​​​​​一&#xff1a;股票预测数据分析 1-1 导入需要使用到的库 import numpy as np import pandas as pd import matplotlib.pyplot as plt import seaborn as sns 1-2 读取csv数据…

Xray正向及反向漏洞检测

文章目录软件介绍下载地址安装教程使用教程主动扫描被动扫描软件介绍 xray 是从长亭洞鉴核心引擎中提取出的社区版漏洞扫描神器&#xff0c;支持主动、被动多种扫描方式&#xff0c;自备盲打平台、可以灵活定义 POC&#xff0c;功能丰富&#xff0c;调用简单&#xff0c;支持 …

Vue3安装scss教程

在开发新项目的时候打算使用scss&#xff0c;新项目使用的是vue3框架&#xff0c;所以发一下scss安装的教程 scss是sass的超集&#xff0c;其实可以说是安装sass Vue3安装scss教程安装依赖vite.config.js配置使用安装依赖 npm install node-sass -S npm install sass-loader …

excel函数应用技巧:那些名不副实的函数列举

对于在职场中徜徉于数据海的Exceler们来说&#xff0c;一提到效率&#xff0c;都会对Excel函数赞不绝口&#xff0c;璀璨耀眼者如VLOOKUP、SUMIF、SUMPRODUCT等&#xff0c;皆是查询统计数据的大杀器&#xff01;但是Excel函数家族成员众多&#xff0c;而且各函数问世时间各不相…

Linux中Docker安装RabbitMQ

1. RabbitMQ介绍 RabbitMQ是一套开源&#xff08;MPL&#xff09;的消息队列服务软件&#xff0c;是由 LShift 提供的一个 Advanced Message Queuing Protocol (AMQP) 的开源实现&#xff0c;由以高性能、健壮以及可伸缩性出名的 Erlang 写成。 2. MQ介绍 MQ是一个互联网架构…

一文带你梳理HTTP的发展历程

HTTP是什么&#xff1f; 首先&#xff0c;HTTP是什么&#xff1f; HTTP 是一种 超文本传输协议(Hypertext Transfer Protocol) 是基于_TCP/IP_的应用层通信协议&#xff0c;它是客户端和服务器之间相互通信的标准。它规定了如何在互联网上请求和传输内容。 你在这里已经可以…