文件操作(详解)

该片博客有点长大家可以通过目录选择性阅读

这是个人主页

敲上瘾-CSDN博客

目录

1. 为什么使⽤⽂件?

2. 什么是⽂件?

2.1 程序⽂件

2.2 数据⽂件

2.3 ⽂件名

3. ⼆进制⽂件和⽂本⽂件?

4. ⽂件的打开和关闭

4.1 流和标准流

4.1.1 流

4.1.2 标准流

4.2 ⽂件指针

4.3 ⽂件的打开和关闭

 5.文件的顺序读写

5.1 fputc函数

5.2 fputs函数

5.3 fgetc函数

5.4 fgets函数 

5.5 fprintf函数

5.6 fscanf函数

​编辑

5.7 fwrite

​编辑

5.8 fread

6.sprintf函数

7.sscanf函数

8.文件的随机读写:

8.1 fseek函数

8.2 ftell函数

8.3 rewind函数

9.错误检测 

9.1 feof函数

9.2 ferror函数

10.缓冲区

10.1文件缓冲区


1. 为什么使⽤⽂件?

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

2. 什么是⽂件?

磁盘(硬盘)上的⽂件是⽂件。
在程序设计中的⽂件有两种:程序⽂件数据⽂件(从⽂件功能的⻆度来的)。

2.1 程序⽂件

程序⽂件包括源程序⽂件(后缀为.c),⽬标⽂件(windows环境后缀为.obj),可执⾏程序(windows
环境后缀为.exe)。

2.2 数据⽂件

⽂件的内容不⼀定是程序,⽽是程序运⾏时读写的数据,⽐如程序运⾏需要从中读取数据的⽂件,或者输出内容的⽂件。
有时候我们会把信息输出到磁盘上,当需要的时候再从磁盘上把数据读取到内存中使⽤,这⾥处
理的就是磁盘上⽂件。

2.3 ⽂件名

⼀个⽂件要有⼀个唯⼀的⽂件标识,以便⽤⼾识别和引⽤。
⽂件名包含3部分:⽂件路径+⽂件名主⼲+⽂件后缀
例如: c:\code\test.txt
为了⽅便起⻅,⽂件标识常被称为⽂件名。

3. ⼆进制⽂件和⽂本⽂件?

根据数据的组织形式,数据⽂件被称为⽂本⽂件或者⼆进制⽂件
数据在内存中以⼆进制的形式存储,如果不加转换的输出到外存的⽂件中,就是⼆进制⽂件。
如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的⽂件就是⽂本⽂件。
⼀个数据在⽂件中是怎么存储的呢?
字符⼀律以ASCII形式存储,数值型数据既可以⽤ASCII形式存储,也可以使⽤⼆进制形式存储。
如有整数10000,如果以ASCII码的形式输出到磁盘,则磁盘中占⽤5个字节(每个字符⼀个字节),⽽⼆进制形式输出,则在磁盘上只占4个字节(VS2019测试)。

4. ⽂件的打开和关闭

4.1 流和标准流

4.1.1 流

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

4.1.2 标准流

那为什么我们从键盘输⼊数据,向屏幕上输出数据,并没有打开流呢?
那是因为C语⾔程序在启动的时候,默认打开了3个流:
stdin-标准输⼊流,在⼤多数的环境中从键盘输⼊,scanf函数就是从标准输⼊流中读取数据。
stdout-标准输出流,⼤多数的环境中输出⾄显⽰器界⾯,printf函数就是将信息输出到标准输出
流中。
stderr-标准错误流,⼤多数环境中输出到显⽰器界⾯。
这是默认打开了这三个流,我们使⽤scanf、printf等函数就可以直接进⾏输⼊输出操作的。
stdin、stdout、stderr三个流的类型是: FILE* ,通常称为⽂件指针
C语⾔中,就是通过 FILE* 的⽂件指针来维护流的各种操作的。

4.2 ⽂件指针

缓冲⽂件系统中,关键的概念是“⽂件类型指针”,简称“⽂件指针”。
每个被使⽤的⽂件都在内存中开辟了⼀个相应的⽂件信息区,⽤来存放⽂件的相关信息(如⽂件的名字,⽂件状态及⽂件当前的位置等)。这些信息是保存在⼀个结构体变量中的。该结构体类型是由系统声明的,取名FILE.
例如,VS2013编译环境提供的 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*的指针变量:
        FILE* pf;//⽂件指针变量
定义pf是⼀个指向FILE类型数据的指针变量。可以使pf指向某个⽂件的⽂件信息区(是⼀个结构体变
量)。通过该⽂件信息区中的信息就能够访问该⽂件。也就是说,通过⽂件指针变量能够间接找到与
它关联的⽂件

4.3 ⽂件的打开和关闭

⽂件在读写之前应该先打开⽂件,在使⽤结束之后应该关闭⽂件。
在编写程序的时候,在打开⽂件的同时,都会返回⼀个FILE*的指针变量指向该⽂件,也相当于建⽴了
指针和⽂件的关系。
ANSI C 规定使⽤ fopen 函数来打开⽂件, fclose 来关闭⽂件。

​
int main()
{
	FILE* pf = fopen("text.txt", "w");//"w"表示以读的方式打开
	fclose(pf);//关闭
	return 0;
}

​

 5.文件的顺序读写

相关函数:

  

5.1 fputc函数

fputc的参数如下

它的功能是把字符character输出到stream指向的文件中,字符的本质就是它的ascll值所以这里用int类型接收。如果写入成功则返回写入的字符的ascll码,失败则返回-1(即EOF)

示例代码:

#include<stdio.h>
#include<errno.h>
int main()
{
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror(fopen);
		return 1;
	}
	for (char i = 'a'; i <= 'z'; i++)
	{
		fputc(i, pf);
	}
	fclose(pf);
	return 0;
}

注意:当不存在test.txt这个文件的时候,以写的形式打开(即"w")程序执行后会创建一个test.txt文件,但如果以读的形式打开(即"r")的话程序会报错 。

注意:如果test.txt这个文件存在的时候,以写的形式打开(即"w")程序执行后会把原有的数据清空进行写入。

注意:不能以读的形式打开然后去写,也不能以写的形式打开然后去读。

一下方式可以找到这个被读写文件

5.2 fputs函数

fputs参数如下:

fputs的功能是把字符串str输出到stream指向的文件中,如果写入成功返回一个非负整数,写入失败,则返回-1(即EOF)。

示例代码:

int main()
{
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror(fopen);
		return 1;
	}
	fputs("1234567", pf);
	char s[] = "abcdef";
	fputs(s, pf);
	fclose(pf);
	return 0;
}

5.3 fgetc函数

fgetc的参数如下:

它的作用是返回stream对应的文件里面的一个字符(它的返回值),所以需要一个字符变量来接收,当读取失败时返回-1,虽然说是返回字符,但fgetc函数的返回类型是int,不过这没关系每个字符都有对应的ascll码值,它的ascll是int类型,这里返回类型用int接收也是为了对应当读取失败时返回的-1,每读取一次stream指向的文件内容,文件指针往后移动一位

示例代码

#include<stdio.h>
int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (!pf)
	{
		perror(fopen);
		return 1;
	}
	char c;
	while ((c = fgetc(pf))!=-1)
	{
		printf("%c", c);
	}
	fclose(pf);
	return 0;
}

5.4 fgets函数 

fgets参数如下:

fgets函数的功能是把stream指向的文件中的num个字符(即一个字符串)输入到str中,返回值是该字符串首字符的地址,当读取失败时返回NULL,每读取一次stream指向的文件内容,stream往后移动num位。

#include<stdio.h>
int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror(fopen);
		return 1;
	}
	char s[30] = { 0 };
	while(fgets(s,2,pf)!=NULL)
	{
		printf(s);
	}
	return 0;
}

5.5 fprintf函数

参数如下:

它的功能是把数据以格式化的形式输出到stream指向的文件

示例代码

#include<stdio.h>
#include<errno.h>
int main()
{
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror(fopen);
		return 1;
	}
	int a = 6;
	char s[10] = "pupm";
	fprintf(pf, "%d %s", a, s);
	fclose(pf);
	return 0;
}

5.6 fscanf函数

参数如下:

它的作用是把stream指向的文件的内容格式化的输入到内存中。

示例代码

#include<stdio.h>
#include<errno.h>
int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror(fopen);
		return 1;
	}
	int a = 0;
	char s[10] = {0};
	fscanf(pf,"%d %s", &a, s);
	printf("%d %s", a,s);
	fclose(pf);
	return 0;
}

5.7 fwrite

fwrite的功能是把ptr指向的空间中的count个元素以二进制的形式写入stream所指向的文件中,其中参数size表示ptr指向的空间中的一个元素占的字节数,返回成功写入的元素个数

注意:文件要以"wb"的形式打开,表示以二进制形式写入。 

示例代码

int main()
{
	FILE* pf = fopen("test.txt", "wb");
	if (pf == NULL)
	{
		perror(fopen);
		return 1;
	}
	int arr[10] = {1,2,3,4,5,6,7,8,9,10};
	fwrite(arr, sizeof(arr[0]), 8, pf);
	fclose(pf);
	return 0;
}

因为是以二进制的形式写进去的,所以里面的内容是看不懂的 

5.8 fread

参数如下:

与fwrite函数相反该函数的功能是把stream所指向的文件内容的count个元素以二进制的形式读取到ptr所指向的空间,其中参数size表示ptr指向的空间中的一个元素占的字节数,返回成功读取到的元素个数

 注意:文件要以"rb"的形式打开,表示以二进制形式读取。

示例代码

int main()
{
	FILE* pf = fopen("test.txt", "rb");
	if (pf == NULL)
	{
		perror(fopen);
		return 1;
	}
	int arr[10] = { 0 };
	int i = 0;
	while (fread(&arr[i], sizeof(arr[0]), 1, pf) == 1)
	{
		printf("%d ", arr[i++]);
	}
	fclose(pf);
	return 0;
}

6.sprintf函数

参数如下:

功能是把格式的数据转化为字符串str

示例代码

#include<stdio.h>
int main()
{
	char s[10] = { 0 };
	int a = 42;
	char c='a';
	sprintf(s, "%d %c", a, c);
	printf("%s", s);
	return 0;
}

7.sscanf函数

sscanf的功能是在字符串s中数据以格式化的形式读取。

示例代码

#include<stdio.h>
int main()
{
	char s[10] = "4k";
	int a;
	char c;
	sscanf(s, "%d%c", &a, &c);
	printf("%d %c", a, c);
	return 0;
}

8.文件的随机读写:

相关函数

fseek     根据⽂件指针的位置和偏移量来定位⽂件指针

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

rewind   让⽂件指针的位置回到⽂件的起始位置

8.1 fseek函数

该函数的作用是文件指针stream指向距origin(起始位置)偏移量为offset的位置 

origin有3中选择:SEEK_SET,SEEK_CUR,SEEK_END三种

 示例代码

int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror(fopen);
		return 0;
	}
	char a, b, c;
	fseek(pf, 5, SEEK_SET);
	a = fgetc(pf);
	printf("%c ", a);

	fseek(pf, 2, SEEK_CUR);
	b = fgetc(pf);
	printf("%c ", b);

	fseek(pf, -2, SEEK_END);
	c = fgetc(pf);
	printf("%c ", c);
	fclose(pf);
	return 0;
}

 test.txt文件内容

运行结果:

8.2 ftell函数

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

 示例代码

int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror(fopen);
		return 0;
	}
	fseek(pf, -1, SEEK_END);
	int ret = ftell(pf);
	printf("%d", ret);
	fclose(pf);
	return 0;
}

8.3 rewind函数

这个函数功能是让文件指针回到文件的起始位置

 示例代码

int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror(fopen);
		return 0;
	}
	fseek(pf, -1, SEEK_END);
	printf("%c ", fgetc(pf));
	rewind(pf);
	printf("%c ", fgetc(pf));
	fclose(pf);
	return 0;
}

9.错误检测 

为什么要有错误检查呢

fgetc 如果读取正常,返回的是读取到字符的ascIl码值,如果读取的过程中遇到文件末尾,或者发生错误,就返回EOF(-1)

fgets如果读取正常,返回的是存储读取到的字符串的字符数组的地址,如果读取的过程中遇到文件末尾,或者发生错误,返回NULL

所以我们是无法知道是因为什么原因文件读取结束的。

9.1 feof函数

该函数用于判断当⽂件读取结束的时候,判断是读取结束的原因是否是:遇到⽂件尾结束(即是否是正常结束)

当是因为遇到⽂件尾而读取结束时返回非0值,否则返回0

9.2 ferror函数

该函数判断读取文件的时候是否发生错误

读取文件的时候是否发生错误0,否则返回非0值。

 示例代码

#include<stdio.h>
#include<errno.h>
int main()
{
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror(fopen);
		return 1;
	}
	char c;
	while ((c = fgetc(pf)) != EOF)
	{
		printf("%c", c);
	}
	if (feof(pf))
	{
		printf("\n遇到文件末尾,读取正常结束\n");
	}
	else if (ferror(pf))
	{
		perror(fgetc);
	}
	fclose(pf);
	return 0;
}

10.缓冲区

在认识缓冲区之前我们先想象一个场景,假设一辆公交车在司机在候车区看到1个人需要乘车就马上拉走一个,拉完再回来拉,每次几乎只拉一个人而路途又很长,那样的话可以想象效率是多么的低,解决方法就是公交车会在候车区停留30分钟到40分钟,等乘客足够多的时候一次性拉走,这样一来效率就高了很多。

像fgetc这样的函数是怎么实现把磁盘上的数据输入到内存呢,其实是依赖于操作系统提供的一些接口,fgetc,fgets,printf,sacnf......这些函数是调用了操作系统指令才得以实现的,如果每读写1个数据都调用一次操作系统,操作系统在执行的时候会被频繁的打断,效率变得很低。所以有了缓冲区的出现,让需要调用操作系统的数据先放在缓冲区,到最后一次性的执行。

数据:乘客

缓冲区:候车区

操作系统:公交车

10.1文件缓冲区

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

注意:关闭文件起到了刷新缓冲区的作用,这也是打开的文件一定要关闭的原因之一。

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

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

相关文章

29-控制流(下):iam-apiserver服务核心功能实现讲解

我们再来看下 iam-apiserver 中的核心功能实现。 这些关键代码设计分为 3 类&#xff0c;分别是应用框架相关的特性、编程规范相关的特性和其他特性。 应用框架相关的特性 应用框架相关的特性包括三个&#xff0c;分别是优雅关停、健康检查和插件化加载中间件。 优雅关停 …

二维码:技术、商业与未来

title: 二维码&#xff1a;技术、商业与未来 date: 2024/4/3 19:12:28 updated: 2024/4/3 19:12:28 tags: 二维码技术商业应用移动支付物联网AR/VR融合智能家居数字化社会 第一章&#xff1a;引言 1. 二维码在数字化时代的重要性和普及程度 在数字化时代&#xff0c;二维码作…

程序员的升级打怪之路

#程序人生 写在前面 转眼间&#xff0c;我已经进入程序员的大门已经近4个春秋了&#xff08;算上实习的话&#xff0c;那就是快5年了…&#x1f436;.&#x1f436;.&#x1f436;不能再展开了&#xff0c;再不就暴露年龄了&#x1f605;&#xff09;。 这段时间&#xff0c…

element-ui card 组件源码分享

今日简单分享 card 组件源码&#xff0c;主要从以下两个方面&#xff1a; 一、card 组件页面结构 二、card 组件属性 2.1 header 属性&#xff0c;设置 header&#xff0c;也可以通过 slot#header 传入 DOM&#xff0c;类型 string&#xff0c;无默认值。 组件使用部分&#…

[做cpu] 第二次仿真实验

实现ori指令后&#xff0c;还得解决流水中数据相关的事&#xff0c;MIPS中只需要解决RAW&#xff08;在写操作后读&#xff09;&#xff0c;利用数据前推解决 相隔两条指令&#xff0c; 通过标志位判断直接把回写的内容作为读入译码的数据。 仿真出错原因&#xff1a;在顶层模…

spring总结-基于XML管理bean超详细

spring ioc总结-基于XML管理bean 前言实验一 [重要]创建bean1、目标和思路①目标②思路 2、创建Maven Module3、创建组件类4、创建spring配置文件7、无参构造器8、用IOC容器创建对象和自己建区别 实验二 [重要]获取bean1、方式一&#xff1a;根据id获取2、方式二&#xff1a;根…

20.安全性测试与评估

每年都会涉及&#xff1b;可能会考大题&#xff1b;多记&#xff01;&#xff01;&#xff01; 典型考点&#xff1a;sql注入、xss&#xff1b; 从2个方面记&#xff1a; 1、测试对象的功能、性能&#xff1b; 2、相关设备的工作原理&#xff1b; 如防火墙&#xff0c;要了解防…

redis---主从复制

主从复制是指将一台redis服务器的数据复制到其他redis服务器&#xff0c;也叫主节点和从节点。 一个主节点可以有多个从节点。而每个从节点只能有一个主节点。数据的复制是单向的&#xff0c;只能由主节点到从节点。一般来说&#xff0c;主节点负责写操作&#xff0c;从节点负…

公众号搜索被降权后多久能恢复?

公众号搜索被降权后的恢复时间是一个复杂的问题&#xff0c;它涉及到多种因素的综合考量。首先&#xff0c;违规的严重程度是一个重要的因素。如果违规行为较为轻微&#xff0c;可能只需要较短的时间就能恢复搜索权重;而如果违规行为较为严重&#xff0c;可能需要更长的时间&am…

vue实现导出列表为xlsx文件

1.安装依赖 npm install --save xlsx file-saver 2.引入依赖 import FileSaver from file-saver; import * as XLSX from xlsx; 3.代码实现 <el-button type"primary" click"exportData">导出数据</el-button><el-tableid"table_ex…

与 ChatGPT 对话

原文&#xff1a;Conversing With ChatGPT 译者&#xff1a;飞龙 协议&#xff1a;CC BY-NC-SA 4.0 ChatGPT 人工智能 尽管人工智能带来了许多好处和进步&#xff0c;但仍有一些话题引发担忧并引发道德、社会和存在问题。以下是与人工智能相关的一些最可怕的话题&#xff1a…

数据结构算法题(力扣)——链表

以下题目建议大家先自己动手练习&#xff0c;再看题解代码。这里只提供一种做法&#xff0c;可能不是最优解。 1. 移除链表元素&#xff08;OJ链接&#xff09; 题目描述&#xff1a;给一个链表的头节点 head 和一个整数 val &#xff0c;删除链表中所有满足值等于 val 的节点…

954: 单链表的链接

学习版 【c语言】 【C艹】 #include<iostream>class LinkedList { public:struct LinkedNode {char val;LinkedNode* next;LinkedNode(char val) :val(val), next(NULL) {};};LinkedList(){dummyHead new LinkedNode(0);tail dummyHead;}~LinkedList() {while (dummy…

初探STM32f407VET6

一、买到了板子&#xff0c;自己分析引脚功能 我在某宝上买到一块stm32f407vet6的板子&#xff0c;图便宜&#xff0c;结果遇上了个态度差的客服。没有说明&#xff0c;没有资料。不能退换&#xff0c;只能自己想办法分析引脚 在嘉里创找到了芯片原理图&#xff08;LQFP-100封…

【智能排班系统】快速消费线程池

文章目录 线程池介绍线程池核心参数核心线程数&#xff08;Core Pool Size&#xff09;最大线程数&#xff08;Maximum Pool Size&#xff09;队列&#xff08;Queue&#xff09;线程空闲超时时间&#xff08;KeepAliveTime&#xff09;拒绝策略&#xff08;RejectedExecutionH…

Python学习笔记-Flask接收post请求数据并存储数据库

1.引包 from flask import Flask, request, jsonify from flask_sqlalchemy import SQLAlchemy 2.配置连接,替换为自己的MySQL 数据库的实际用户名、密码和数据库名 app Flask(__name__) #创建应用实列 app.config[SQLALCHEMY_DATABASE_URI] mysqlpymysql://ro…

微软文本转语音和语音转文本功能更新,效果显著!

今天我要和大家分享一个新功能更新——微软的文本转语音和语音转文本功能。最近&#xff0c;微软对其AI语音识别和语音合成技术进行了重大升级&#xff0c;效果非常好&#xff0c;现在我将分别为大家介绍这两个功能。 先来听下这个效果吧 微软文本转语音和语音转文本功能更新 …

二分答案(砍树,借教室)

二分的两种情况附代码&#xff1a; 二分查找条件&#xff1a;单调&#xff0c;二段性 例题1&#xff1a;P1873 [COCI 2011/2012 #5] EKO / 砍树 - 洛谷 | 计算机科学教育新生态 (luogu.com.cn) 上代码&#xff1a; #include<bits/stdc.h> using namespace std; const …

【数据结构与算法】归并排序(详解:递归与非递归的归并排序 | 赠:冒泡排序和选择排序)

前言 本篇博客会对排序做一个收尾&#xff0c;将最经典的七大排序介绍完毕。 这次的重点正如标题&#xff0c;主要讲的是归并排序&#xff0c;还会带过相对简单很多的冒泡排序和选择排序。在最后还会给这七大排序做出一个时间复杂度和稳定性展示的总结收尾。同时&#xff0c;这…

钉钉事件订阅前缀树算法gin框架解析

当钉钉监测到发生一些事件&#xff0c;如下图 此处举例三个事件user_add_org、user_change_org、user_leave_org&#xff0c;传统的做法是&#xff0c;我们写三个if条件&#xff0c;类似下图 这样字符串匹配效率比较低&#xff0c;于是联想到gin框架中的路由匹配算法&#xff0…