C语言之文件操作(万字详解)

个人主页(找往期文章包括但不限于本期文章中不懂的知识点): 我要学编程(ಥ_ಥ)-CSDN博客

目录

前言

文件的打开和关闭 

流和标准流

文件指针

文件的打开和关闭 

文件的顺序读写 

顺序读写函数介绍

fputc的使用 

fgetc的使用 

fputs的使用 

fgets的使用 

fprintf的使用 

fscanf的使用 

对比一组函数(输入与输出) 

sprintf的使用 

sscanf的使用

fwrite的使用 

fread的使用 

文件的随机读写

fseek的使用

ftell的使用 

rewind的使用 

文件读取结束的判定

被错误使用的 feof

文件缓冲区 


前言

为什么要使用文件? 

我们写的程序的数据是存储在电脑的内存中,如果没有文件,程序退出,内存回收,数据就丢失 了,等再次运行程序,是看不到上次程序的数据的,如果要将数据进行持久化的保存,我们就要使用文件。

什么是文件?

磁盘(硬盘)上的文件是文件。 但是在程序设计中,我们一般谈的文件有两种:程序文件、数据文件(从文件功能的角度来分类的)。

程序文件包括源程序文件(后缀为.c),目标文件(windows环境后缀为.obj),可执行程序(windows 环境后缀为.exe)。

数据文件的内容不一定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件,或者输出内容的文件。

本文讨论的是数据文件。 在以前所处理数据的输入输出都是以终端为对象的,即从终端的键盘输入数据,运行结果显示到显示器上。 其实有时候我们会把信息输出到磁盘上,当需要的时候再从磁盘上把数据读取到内存中使用,这里处理的就是磁盘上的文件。

文件名 :一个文件要有一个唯一的文件标识,以便用户识别和引用。 文件名包含3部分:文件路径+文件名主干+文件后缀 例如: c:\code\test.txt 为了方便起见,文件标识常被称为文件名。即一个文件要有一个唯一的文件名。

二进制文件和文本文件 根据数据的组织形式,数据文件被称为文本文件或者二进制文件。 数据在内存中以二进制的形式存储,如果不加转换的输出到外存(与之对应的就是内存)的文件中,就是二进制文件。 如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII码字符的形式存储的文件就是文本文件。 一个数据在文件中是怎么存储的呢? 字符一律以ASCII码形式存储,数值型数据既可以用ASCII码形式存储,也可以使用二进制形式存储。 如整数10000,如果以ASCII码的形式输出到磁盘,则在磁盘中占用5个字节(每个字符一个字节,'1' '0' '0' '0' '0' '0'),而以二进制形式输出,则在磁盘上只占4个字节(一个整数4个字节)。

文件的打开和关闭 

流和标准流

我们程序的数据需要输出到各种外部设备,也需要从外部设备获取数据,不同的外部设备的输入输出操作各不相同,为了方便程序员对各种设备进行方便的操作,我们抽象出了流的概念,我们可以把流想象成流淌着字符的河。 C程序针对文件、画面、键盘等的数据输入输出操作都是通过流操作的。 一般情况下,我们要想向流里写数据,或者从流中读取数据,都是要打开流,然后操作。

标准流 那为什么我们使用 scanf函数 从键盘输入数据,使用 printf函数 向屏幕上输出数据,并没有打开流呢? 那是因为C语言程序在启动的时候,默认打开了3个流:

• stdin - 标准输入流,在大多数的环境中从键盘输入,scanf函数就是从标准输入流中读取数据。

• stdout - 标准输出流,大多数的环境中输出至显示器界面,printf函数就是将信息输出到标准输出流中。

• stderr - 标准错误流,大多数环境中输出到显示器界面。

这是默认打开了这三个流,我们使用scanf、printf等函数就可以直接进行输入输出操作的。 stdin、stdout、stderr 三个流的类型是: FILE* ,通常称为文件指针。 C语言中,就是通过 FILE* 的文件指针来维护流的各种操作的。

文件指针

缓冲文件系统中,关键的概念是“文件类型指针”,简称“文件指针”。 每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。这些信息是保存在一个结构体变量中的。该结构体类型是由系统声明的,取名 FILE。

VS 编译环境提供的 stdio.h 头文件中有以下的文件类型申明: 

struct _iobuf {
	char* _ptr;
	int _cnt;
	char* _base;
	int _flag;
	int _file;
	int _charbuf;
	int _bufsiz;
	char* _tmpfname;
};
typedef struct _iobuf FILE;

不同的C编译器的FILE类型包含的内容不完全相同,但是大同小异。 每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息,使用者不必关心细节。 一般都是通过一个FILE的指针来维护这个FILE结构的变量,这样使用起来更加方便。

FILE* pf;//⽂件指针变量

定义pf是一个指向FILE类型数据的指针变量。可以使pf指向某个文件的文件信息区(是一个结构体变量)。通过该文件信息区中的信息就能够访问该文件。也就是说,通过文件指针变量能够间接找到与它关联的文件。 就比如下面这幅图:

文件的打开和关闭 

文件在进行读写的操作之前应该先打开文件,在使用结束之后应该关闭文件。 在编写程序的时候,打开文件的同时,都会返回一个FILE*的指针变量指向该文件,也相当于建立了指针和文件的关系。 ANSI C(标准C) 规定使用 fopen 函数来打开文件, fclose 来关闭文件。函数原型如下:

//打开⽂件
//filename——>文件名
//mode——>方式(以什么样的方式打开这个文件,例如:读,写……)
FILE * fopen ( const char * filename, const char * mode );


//关闭⽂件
//stream——>就是一个文件指针(要被关闭的文件)
int fclose ( FILE * stream );

在介绍这两个函数之前,我们还需要学习一个函数 perror 。

这个函数是用来打印错误信息的。这个函数与我们学过的 strerror 函数有关。strerror 函数是把错误码对应的错误信息的地址传给一个char*的指针;如果我们再用 printf 打印的话,就可以知道其内容了,而这个perror就是直接一步到位,打印错误信息。

#include <stdio.h>
#include <string.h>
int main()
{
	char* ret = strerror(1);
	printf("%s\n", ret);
	perror("1");
	return 0;
}

当然这里因为没有错误信息,所以它就打印了没有错误。perror 就是会把传进去的参数打印,然后在后面加上冒号与空格。注意的是这个参数传进去的时候要加双引号。

现在就来学习fopen 函数。下面这个表格是文件的打开方式分类:

文件打开方式含义如果指定文件不存在 
 "r"(只读)为了输入数据,打开一个已经存在的文本文件出错
"w"(只写)为了输出数据,打开一个文本文件建立一个新的文件
"a"(追加)向文本文件末尾添加数据建立一个新的文件
"rb"(只读)为了输入数据,打开一个二进制文件出错
"wb"(只写)为了输出数据,打开一个二进制文件建立一个新的文件
"ab"(追加)向一个二进制文件末尾添加数据建立一个新的文件
"r+"(读写)为了读和写,打开一个文本文件出错
"w+"(读写)为了读和写,建立一个新的文件建立一个新的文件
"a+"(读写)打开一个文件,在文件末尾进行读写建立一个新的文件
"rb+"(读写)为了读和写打开一个二进制文件出错
"wb+"(读 写)为了读和写,新建一个新的二进制文件建立一个新的文件
"ab+"(读 写)打开一个二进制文件,在文件末尾进行读和写建立一个新的文件
#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "r");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;//打开失败就不需要往后进行了
	}
	//打开成功
	// ……
	//后续操作
	return 0;
}

注意如果没有写绝对路径的话,系统就只会在本文件下寻找(就是与这个 .c 文件在同一文件下)。绝对路径就是:D:\c语言\writing-code-for-blogs\test_3_12\test_3_12 这种。当然为了防止有转义字符的出现就需要加上双斜杠。

#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "w");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;//打开失败就不需要往后进行了
	}
	//打开成功
	// ……
	//后续操作
	return 0;
}

文件的顺序读写 

顺序读写函数介绍

函数名功能适用于
fgetc字符输入函数所有输入流
fputc字符输出函数所有输出流
fgets文本行输入函数所有输入流
fputs文本行输出函数所有输出流
fscanf格式化输入函数所有输入流
fprintf格式化输出函数所有输出流
fread二进制输入文件
fwrite二进制输出文件

上面说的适用于所有输入流一般指适用于标准输入流和其他输入流(如文件输入流);所有输出流一般指适用于标准输出流和其他输出流(如文件输出流)。 现在我们就可以往文件里写入一些字符了。

fputc的使用 

fputc 就是把字符输出到流中。 

#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "w");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;//打开失败就不需要往后进行了
	}
	//打开成功,往其中写入一些字符
	fputc('a', pf);
	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

如果要写多个就得多次调用这个fputc 函数。

#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "w");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;//打开失败就不需要往后进行了
	}
	//打开成功,往其中写入一些字符
	for (int i = 0; i < 26; i++)
	{
		fputc('a' + i, pf);
	}
	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

如果我们要打印到屏幕上,就写stdout就可以了。

fgetc的使用 

这个fgetc就是从所有流中获取字符,也就是读取。

#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "r");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;//打开失败就不需要往后进行了
	}
	//打开成功,读取一些字符
	//读取成功返回字符的ASCll码值,读取失败或者遇到文件末尾返回EOF
	int ret = fgetc(pf);
	printf("%d\n", ret);
	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "r");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;//打开失败就不需要往后进行了
	}
	//打开成功,读取一些字符
	//读取成功返回字符的ASCll码值,读取失败或者遇到文件末尾返回EOF
	int ret = fgetc(stdin);
	printf("%d\n", ret);
	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

fputs的使用 

fputs与fputc类似。 

#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "w");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;//打开失败就不需要往后进行了
	}
	//打开成功,输出一些字符
	fputs("hello world", pf);
	fputs("I want learn", pf);
	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

fgets的使用 

fgets与fgetc就有一些不一样了。

fgets是把从stream中读取的num-1个字符放到str这个字符指针中。

#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "r");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;//打开失败就不需要往后进行了
	}
	//打开成功,输出一些字符
	char str[10] = { 0 };
	fgets(str, 8, pf);
	printf("%s\n", str);
	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

当要读取的字符数超过总字符时,就会只读取总字符数;又或者当读取时遇到'\n'(换行符),也会停止本次读取。

#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "r");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//打开成功,输入一些字符
	char str[10] = { 0 };
	fgets(str, 8, pf);
	printf("%s", str);
	fgets(str, 8, pf);
	printf("%s", str);
	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

fprintf的使用 

这个fprintf与printf相比,就只多了前面那个FILE* stream参数,我们就可以模拟printf来实现fprintf函数。  

struct S
{
	int i;
	float f;
};
#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "w");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	struct S s = { 10, 3.14f };
	//打开成功,输入一些字符
	
	//printf("%d %f\n", s.i, s.f);
	fprintf(pf, "%d %f\n", s.i, s.f);

	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

可以从结果来看:fprintf是把s.i , s.f 按照按照%d %f的格式打印到pf中。 

fscanf的使用 

这个fscanf与scanf相比,就只多了前面那个FILE* stream参数,我们就可以模拟scanf来实现fscanf函数。 

struct S
{
	int i;
	float f;
};
#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "r");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	struct S s = { 0 };

	//scanf("%d%f", &(s.i), &(s.f));
	fscanf(pf, "%d%f", &(s.i), &(s.f));

	printf("%d %f\n", s.i, s.f);//看看是否读取成功
	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

 可以从结果来看:fscanf是把pf中的数据按照按照%d %f的格式输入到s.i , s.f 中。

对比一组函数(输入与输出) 

scanf ——>从标准输入流读取格式化的数据。

printf  ——>从标准输出流写格式化的数据。

fscanf——>适用于所有输入流的格式化输入函数。

fprintf——>适用于所有输出流的格式化输出函数。

sscanf——>从字符串中读取格式化的数据

sprintf——>将格式化的数据转换成字符串

sprintf的使用 
struct S
{
	int i;
	float f;
};
#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "w");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	struct S s = { 10 ,3.14f };
	char arr[20] = { 0 };
	//printf("%d %f\n", s.i, s.f);
	sprintf(arr, "%d %f\n", s.i, s.f);
	
	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

与printf相比,也只是多了一个字符指针参数。

ssprintf也是把s.i, s.f按照%d %f的格式放到arr中。 

sscanf的使用
struct S
{
	int i;
	float f;
};
#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "r");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	struct S s = { 10 ,3.14f };
	struct S tmp = { 0 };
	char arr[20] = { 0 };

	sprintf(arr, "%d %f\n", s.i, s.f);
	//scanf("%d%f", &(s.i), &(s.f));
	sscanf(arr, "%d%f", &(tmp.i), &(tmp.f));

	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

sscanf是把arr中数据按照%d %f的格式放到&(tmp.i) , &(tmp.f)。

sprintf函数一般是与sscanf函数一起使用。

上面这些函数都是属于文本操作函数,操作的是文本文件,而接下来我们要学习的是操作二进制文件的函数。

fwrite的使用 

上面这个函数的大概意思是:写count个,大小为size的字节的元素到这个流里(stream),从ptr所指向的这个内存开始写。 

struct S
{
	int i;
	float f;
};
#include <stdio.h>
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "wb");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}

	struct S s = { 10, 3.14f };
	//从&s这个位置开始写,写一个大小为sizeof(s)的数据到pf中
	fwrite(&s, sizeof(s), 1, pf);

	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

如果要把它看懂的话,就得用二进制的方式来读,这就需要用到fread函数了。

fread的使用 

我们会发现这个和fwrite的参数是一样的。其实这个函数与fwrite的功能刚好相反。它是从流中(stream)的数据, 取出count个大小为size的数据放到ptr所指向的那块空间中。并且当fread会返回一个size_t的值,表示读取成功的个数。

#include <stdio.h>
struct S
{
	int i;
	float f;
};
int main()
{
	//打开文件
	FILE* pf = fopen("data.txt", "wb");
	//打开失败
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	struct S s1 = { 10, 3.14f };
	//写入数据
	fwrite(&s1, sizeof(s1), 1, pf);
	fclose(pf);

	pf = fopen("data.txt", "rb");
	struct S s2 = { 0 };
	//从pf中读取一个大小为sizeof(s)的数据放到&s指向的那块空间中
	fread(&s2, sizeof(s2), 1, pf);
	printf("%d %f\n", s2.i, s2.f);//看看是否读取成功
	//关闭文件
	fclose(pf);
	pf = NULL;
	return 0;
}

从输出的结果来看这个我们写入与读取的操作都是正确的。 

文件的随机读写

fseek的使用

根据文件指针的位置和偏移量来定位文件指针。

我们知道当打开一个文件时,文件指针是指向这个文件的起始位置的。如果我们想要读取后面的数据,总不能一个一个的往后读取吧,太不方便了。因此就提供了一个函数fseek,可以让我们从想要的位置开始读取。

参数解析:offset——>偏移量  ; origin——>从哪里开始读取。给我们提供了三个选择:文件的起始位置(SEEK_SET),文件的末尾(SEEK_END),文件指针当前指向的位置(SEEK_CUR) ; stream——>从哪个流开始读取。

#include <stdio.h>
int main()
{
	FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//假设我们这个已经在data.txt这个文件中写入abcdefghi
	int ret = fgetc(pf);
	printf("%c\n", ret);

	fseek(pf, 5, SEEK_SET);

	ret = fgetc(pf);
	printf("%c\n", ret);

	fclose(pf);
	pf = NULL;
	return 0;
}

注意这个偏移量也是由正负之分的。正的,文件指针就往右偏;负的,文件指针就往左偏。

并且我这个SEEK_SET是从文件的起始地址开始的,如果是使用SEEK_CUR,那么就只要偏移4次就够了,因为读取a之后,文件指针就指向b去了。

ftell的使用 

返回文件指针相对于起始位置的偏移量。

#include <stdio.h>
int main()
{
	FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//偏移了次
	fseek(pf, 5, SEEK_SET);
	//相较于起始地址,偏移量是5
	int ret = ftell(pf);
	printf("%d\n", ret);

	fclose(pf);
	pf = NULL;
	return 0;
}

rewind的使用 

让文件指针的位置回到文件的起始位置

#include <stdio.h>
int main()
{
	FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fseek(pf, 5, SEEK_SET);
	int ret = fgetc(pf);
	printf("%c\n", ret);

	rewind(pf);
	ret = fgetc(pf);
	printf("%c\n", ret);
	
	fclose(pf);
	pf = NULL;
	return 0;
}

这些文件都是fseek那个代码里的文件,也就是这个文件里是:abcdefghi。 

文件读取结束的判定

被错误使用的 feof

牢记:在文件读取过程中,不能用feof函数的返回值直接来判断文件的是否结束。

feof 的作用是:当文件读取结束的时候,判断读取结束的原因是否是:遇到文件尾结束。

1. 文本文件读取是否结束,判断返回值是否为 EOF ( fgetc ),或者 NULL ( fgets )

例如: • fgetc 判断是否为 EOF 。 • fgets 判断返回值是否为 NULL 。

2. 二进文件的读取结束判断,判断返回值是否小于实际要读的个数。

• fread判断返回值是否小于实际要读的个数。

文本文件:

#include <stdio.h>
#include <stdlib.h>
int main()
{
	int c; 
	FILE* fp = fopen("test.txt", "r");
	if (!fp)//fp==NULL 
	{
		perror("File opening failed");
		return EXIT_FAILURE;
	}
	//fgetc 当读取失败的时候或者遇到文件结束的时候,都会返回EOF
	while ((c = fgetc(fp)) != EOF) 
	{
		putchar(c);
	}
	//判断是什么原因结束的
    //ferror是用来判断是不是遇到读取错误
	if (ferror(fp))//读取遇到错误
		puts("I/O error when reading");
	else if (feof(fp))//遇到文件末尾
		puts("End of file reached successfully");
	fclose(fp);
	fp = NULL;
	return 0;
}

二进制文件:

#include <stdio.h>
enum { SIZE = 5 };
int main()
{
	double a[SIZE] = { 1.0,2.0,3.0,4.0,5.0 };
	FILE* fp = fopen("test.bin", "wb"); 
	if (fp == NULL)
	{
		perror("fopen");
		return 1;
	}
	//sizeof * a <——> sizeof(*a) <——> 这个也就是a数组的第一个元素
	fwrite(a, sizeof * a, SIZE, fp);//写 double 的数组
	fclose(fp);

	double b[SIZE];
	fp = fopen("test.bin", "rb");
	size_t ret_code = fread(b, sizeof * b, SIZE, fp); // 读 double 的数组
	if (ret_code == SIZE)//全部读取成功
	{
		puts("Array read successfully, contents: ");
		for (int n = 0; n < SIZE; ++n)
			printf("%f ", b[n]);
		putchar('\n');
	}
	else 
	{
		if (feof(fp))//读取错误
		{
			printf("Error reading test.bin: unexpected end of file\n");
		}
		else if (ferror(fp)) //遇到文件末尾
		{
			perror("Error reading test.bin");
		}
	}
	fclose(fp);
	fp = NULL;
	return 0;
}

文件缓冲区 

ANSI C (标准C)标准采用“缓冲文件系统” 处理的数据文件的,所谓缓冲文件系统是指系统自动地在内存中为程序中每一个正在使用的文件开辟一块“文件缓冲区”。从内存向磁盘输出数据会先送到内存中的缓冲区,装满缓冲区后才一起送到磁盘上。如果从磁盘向计算机读入数据,则从磁盘文件中读取数据输入到内存缓冲区(充满缓冲区),然后再从缓冲区逐个地将数据送到程序数据区(程序变量等)。缓冲区的大小根据C编译系统决定的。 

#include <stdio.h>
#include <windows.h>//Sleep函数所需的头文件
int main()
{
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fputs("abcdef", pf);//先将代码放在输出缓冲区
	printf("睡眠10秒-已经写数据了,打开test.txt文件,发现文件没有内容\n");
	Sleep(10000);//休眠10000毫秒
	printf("刷新缓冲区\n");
	fflush(pf);//刷新缓冲区时,才将输出缓冲区的数据写到文件(磁盘)
	printf("再睡眠10秒-此时,再次打开test.txt文件,文件有内容了\n");
	Sleep(10000);
	fclose(pf);
	//注:fclose在关闭文件的时候,也会刷新缓冲区
	pf = NULL;
	return 0;
}

这里可以得出一个结论: 因为有缓冲区的存在,C语言在操作文件的时候,需要做刷新缓冲区或者在文件操作结束的时候关闭文件。 如果不做,可能导致读写文件的问题。

好啦!本期文件操作的学习到此就结束了!下一期,我们再一起学习吧!

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

为什么大口径管材产线中使用大直径测径仪?

关键词&#xff1a;测径仪, 管材测径仪, 外径检测, 大口径管材测径仪, 大口径测径仪 在一个繁忙的钢管制造工厂里&#xff0c;生产线上的钢管正源源不断地被制造出来。然而&#xff0c;工程师们发现了一个问题&#xff1a;由于测量误差&#xff0c;部分钢管的直径不符合规格&am…

【学一点RISC-V】RISC-V IMSIC

IMSIC RISC-V AIA 文档 第三章 Incoming MSI Controller (IMSIC) 传入 MSI 控制器&#xff08;IMSIC&#xff09;是一个可选的 RISC-V 硬件组件&#xff0c;与 hart 紧密相连&#xff0c;每个 hart 有一个 IMSIC。IMSIC 接收并记录 Hart 的传入消息信号中断 (MSI)&#xff0c;并…

P5016 [NOIP2018 普及组] 龙虎斗

1.思路 1.先计算包含天降神兵的势力 #include <iostream> using namespace std; int main(){long long n;cin >> n;long long c[n+10];for (long long i= 1; i <= n; i++){cin >> c[i];} long long m,p1,s1,s2;cin >> m >> p1 >> s1…

C++类和对象一

#include <iostream> using namespace std;//设计一个学生类 class CStudent {public: //公有成员void InputData(){cout << "请输入学号";cin >> sno;cout << "请输入姓名";cin >> sname;cout << "请输入分…

2024年最新腾讯云优惠券获得方法整理

腾讯云作为国内领先的云服务提供商&#xff0c;其优质的产品和服务深受用户喜爱。而腾讯云优惠券则是用户在使用腾讯云服务时能够享受到的一项福利&#xff0c;可以有效降低上云成本。那么&#xff0c;2024年如何获得腾讯云优惠券呢&#xff1f;本文将为大家详细整理最新腾讯云…

U盘启动盘 制作Linux Ubuntu CentOS系统启动盘 系统安装

U盘启动盘 制作Linux Ubuntu CentOS系统启动盘 系统安装 准备条件 准备一个U盘&#xff0c;建议容量至少为8GB&#xff0c;以便存放系统镜像文件 一台已经安装好操作系统的计算机&#xff0c;用于制作U盘启动盘 Ubuntu和CentOS的Linux ISO镜像文件。可以从官方网站或相关资源…

Hadoop大数据应用:NFS网关 连接 HDFS集群

目录 一、实验 1.环境 2.NFS网关 连接 HDFS集群 3. NFS客户端挂载HDFS文件系统 二、问题 1.关闭服务报错 2.rsync 同步报错 3. mount挂载有哪些参数 一、实验 1.环境 &#xff08;1&#xff09;主机 表1 主机 主机架构软件版本IP备注hadoop NameNode &#xff08;…

chatGPT的耳朵!OpenAI的开源语音识别AI:Whisper !

语音识别是通用人工智能的重要一环&#xff01;可以说是AI的耳朵&#xff01; 它可以让机器理解人类的语音&#xff0c;并将其转换为文本或其他形式的输出。 语音识别的应用场景非常广泛&#xff0c;比如智能助理、语音搜索、语音翻译、语音输入等等。 然而&#xff0c;语音…

Leetcode-热题100-持续更新

时间不够&#xff0c;只能背诵哎&#xff0c;没办法&#xff0c;难题不写&#xff0c;简单题和中等题。 def majorityElement(nums):# 使用字典 count 来记录每个元素的出现次数count {}for num in nums:if num in count:count[num] 1else:count[num] 1length len(nums)#…

08-java基础-锁之AQSReentrantLockBlockingQueueCountDownLatchSemapho

文章目录 0&#xff1a;AQS简介-常见面试题AQS具备特性state表示资源的可用状态AQS定义两种资源共享方式AQS定义两种队列自定义同步器实现时主要实现以下几种方法&#xff1a;同步等待队列条件等待队列 1&#xff1a;AQS应用之ReentrantLockReentrantLock如何实现synchronized不…

Jsp在Javaweb中扮演什么角色?

1.什么是Jsp JSP&#xff08;Java Server Pages&#xff0c;Java 服务器页面&#xff09;是一种动态网页技术&#xff0c;它允许在 HTML 页面中嵌入 Java 代码&#xff0c;并由 Web 服务器在请求页面时动态生成 HTML 页面。JSP 通常用于创建动态 Web 内容&#xff0c;如交互式表…

开关电源的线性调整率是什么?怎么检测线性调整率?

开关电源线性调整率 开关电源线性调整率是指输入电压在额定范围内变化时&#xff0c;开关电源输出电压随之变化的比率。线性调整率对开关电源的电压稳定性有着重要影响&#xff0c;通常开关电源的线性调整率在1%~5%之间。线性调整率越小&#xff0c;说明电压越稳定&#xff1b;…

【SpringCloud微服务实战08】RabbitMQ 消息队列

MQ异步通信优缺点: 优点: 吞吐量提升:无需等待订阅者处理完成,响应更快速 故障隔离:服务没有直接调用,不存在级联失败问题 调用间没有阻塞,不会造成无效的资源占用 耦合度极低,每个服务都可以灵活插拔,可替换 流量削峰:不管发布事件的流量波动多大,都由Broker接收,…

网络工程师笔记14

VLAN原理与配置 VLAN 的功能 划分广播域&#xff0c;隔离广播域 VLAN的应用划分 VLAN配置命令

STM32第九节(中级篇):RCC——时钟树讲解(第一节)

目录 前言 STM32第九节&#xff08;中级篇&#xff09;&#xff1a;RCC——时钟树讲解 时钟树主系统时钟讲解 HSE时钟 HSI时钟 锁相环时钟 系统时钟 SW位控制 HCLK时钟 PCLKI时钟 PCLK2时钟 RTC时钟 MCO时钟输出 6.2.7时钟安全系统(CSS&#xff09; 小结 前言 从…

【你也能从零基础学会网站开发】Web建站之javascript入门篇 简单介绍DOM与XML

&#x1f680; 个人主页 极客小俊 ✍&#x1f3fb; 作者简介&#xff1a;程序猿、设计师、技术分享 &#x1f40b; 希望大家多多支持, 我们一起学习和进步&#xff01; &#x1f3c5; 欢迎评论 ❤️点赞&#x1f4ac;评论 &#x1f4c2;收藏 &#x1f4c2;加关注 XML概述 XML是…

windows批处理脚本(cmd指令)

一、简介 最早期的电脑系统是DOS系统&#xff0c;DOS系统只有一个黑漆漆的窗口&#xff0c;需要自己输入命令&#xff0c;所以学习命令是很有必要的&#xff0c;那么CMD命令大全是什么?直到今天的Windows系统&#xff0c;还是离不开DOS命令的操作。如今懂得使用windows批处理脚…

vue解决:this.$refs引用子组件报错 is not a function

我这里有三个可能 可能是你的解决问题的办法&#xff1a; 1.如果你是想跳转到某个页面发现出现此问题&#xff1a; 那是因为你么有导出 2.可能是你的import引入的指定文件的路径有问题 3.你在你跳转的新的页面么有对应的方法

Self-supervised Contextual Keyword and Keyphrase Retrieval with Self-Labelling

文章目录 题目摘要方法数据集实验 题目 通过自我标记进行自我监督的上下文关键字和关键词短语检索 论文地址&#xff1a;https://www.preprints.org/manuscript/201908.0073/v1 项目地址&#xff1a;https://github.com/naister/Keyword-OpenSource-Data 摘要 在本文中&#x…

【SpringCloud微服务实战01】Eureka 注册中心

前言 在 Eureka 架构中&#xff0c;微服务角色有两类&#xff1a; EurekaServer &#xff1a;服务端&#xff0c;注册中心 记录服务信息 心跳监控 EurekaClient &#xff1a;客户端 Provider &#xff1a;服务提供者&#xff0c;例如案例中的 user-service …