C语言 文件操作 深度解析 #重点知识:文件操作函数的使用#

news2024/10/6 6:00:56

文章目录

  • 前言
  • 1. 为什么使用文件
  • 2. 什么是文件
    • 程序文件
    • 数据文件
  • 3. 文件的打开和关闭
  • 4. 文件的顺序读写
    • `fgetc`
    • `fputc`
    • `fgets`
    • `fputs`
    • `fprintf`
    • `fscanf`
    • 流的介绍
    • `fwrite`
    • `fread`
  • 5. 文本文件和二进制文件
  • 6. 文件读取结束的判定
    • 被错误使用的feof
  • 7. 文件缓冲区
  • 写在最后

前言

  • 相信大家对文件都不陌生,我们在计算机的一系列操作都离不开文件,如果Linux没有了文件,那他就不叫Linux了,有了文件,我们的计算机才能够正常运行,有了文件我们才能够在计算机上写东西并能够帮我们保存下来。
  • C语言也提供了一系列的函数来操控文件,本篇文章只介绍那些常见和常用的函数。

1. 为什么使用文件

我们有可能会写通讯录,写过通讯录的都知道,如果没有文件操作,程序运行后输入的数据在程序结束就没有了,联系人信息也就没有了,如果不能存放联系人的信息,那通讯录还有什么用?所以必然的我们需要文件操作来将输入的联系人的信息存放到文件当中,下一次运行时再将之前已经保存在文件中的联系人的信息拿出来,这一系列的操作就是C语言的文件操作,对应的就需要使用一些相关操作函数。简单来说,文件操作能够将输入的通讯录的数据存放在硬盘文件中,这样就做到了数据的持久化。

所以使用文件,只有好处没有坏处。

2. 什么是文件

  • 硬盘上的文件就是文件。
  • 而文件又分为程序文件和数据文件

程序文件

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

数据文件

  • 数据文件的内容不一定是程序,而是程序运行时读写的数据,比如程序运行需要从中读取数据的文件,或者输出内容的文件。
  • 本章讨论的是数据文件。
  • 在前面我们所学习所处理的数据的输入输出都是以终端为对象的,即从终端的键盘输入数据,运行结果显示到显示器上。
  • 其实有时候我们会把信息输出到磁盘上,当需要的时候再从磁盘上把数据读取到内存中使用,这里处理的就是磁盘上文件

3. 文件的打开和关闭

在理解前面的文件介绍后,这里我们开始学习C语言是如何进行文件操作的?

要进行文件操作,当然要知道文件的打开和关闭,在这之前,我们要学习文件指针。

  • 缓冲文件系统中,关键的概念是“文件类型指针”,简称“文件指针”。

  • 每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等)。这些信息是保存在一个结构体变量中的。该结构体类型是有系统声明的,取名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; // 取名FILE
  • 不同的C编译器的FILE类型包含的内容不完全相同,但是大同小异。

  • 每当打开一个文件的时候,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息,我们使用者不必关心细节。

  • 一般都是通过一个FILE的指针来维护这个FILE结构的变量,这样使用起来更加方便。
    例如,这里创建一个FILE*的指针变量:

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

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

如下图:

在这里插入图片描述

有了上面的文件指针相关概念的认知后,接下来就可以进行文件的打开和关闭了

  • 文件的打开和关闭

    1. 文件在读写之前应该先打开文件,在使用结束之后应该关闭文件。
    2. 在编写程序的时候,在打开文件的同时,都会返回一个FILE*的指针变量指向该文件,也相当于建立了指针和文件的关系。
    3. ANSIC 规定使用fopen函数来打开文件,fclose来关闭文件。
  • fopen

在这里插入图片描述
如果打开文件失败,返回一个NULL

  • fclose
int fclose ( FILE * stream );
// 可以直接这样关闭
fclose(pf);  // pf 是一个 FILE* 的指针变量,他此时指向一个文件,这里将他关闭
// 最后这里最好将这个pf指针置空
pf = NULL;

这两个函数的头文件都是<stdio.h>

文件的打开方式有很多,如下图:

在这里插入图片描述

当然,这么多方式,我们只要着重熟练使用前5个就Ok了。

说了这么多,下面实例操作一下:

假设不自己创建文件以读的方式打开文件:

#include <stdio.h>
#include <stdlib.h>

int main()
{
	// 这里读文件,如果此时当前目录下没有这个文件,fopen返回一个空指针
	// 假设此时没有这个文件
	FILE* pf = fopen("test.txt", "r");	
	if (pf == NULL)
	{
		// 没有打开成功
		// 打印错误信息
		perror("open fail");
		exit(-1);
	}
	else
	{
		// 打开成功后
		// ......
		// 一系列文件操作

		// 关闭文件
		fclose(pf);  // 传递指向文件的指针
		pf = NULL;
	}

	return 0;
}

在这里插入图片描述
可以看到,报错信息很明显:没有这个文件,说明需要文件已经存在,这也就对应了上面文件打开方式图的内容,如果此时以写(w)的方式打开?根据上面的内容,他会自动创建一个文件。

打开文件代码改为:

FILE* pf = fopen("test.txt", "w");

此时程序运行,终端什么都没有,但是在当前文件多出了一个文件,正是fopen里的那个test.txt文件:

在这里插入图片描述

所以,以何种形式打开文件,其效果是不一样的,这也给了我们更多的选择。

4. 文件的顺序读写

文件的顺序读写当然需要一批函数来帮助我们操作,下面是一系列这样有这样功能的函数,接下来一一介绍和使用:

在这里插入图片描述

fgetc

fgetc是从文件中读取一个字符的函数

下面是函数参数原型:

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

话不多说,先在当前文件创建一个test.txt,并在里面输入abcdef,然后进行fgetc的文件操作:

#include <stdio.h>  // 对应函数头文件
#include <stdlib.h>

int main()
{
	// fgetc 相当于从文件中读取字符,所以这里要以r的形式打开文件
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror("open fail");
		exit(-1);
	}

	// 打开成功
	// 开始读取字符
	int ch = fgetc(pf);  // 得到字符的ASCLL码值
	printf("%c\n", ch);
	ch = fgetc(pf);  // 得到下一个字符的ASCLL码值,函数会自动往下找
	printf("%c\n", ch);
	ch = fgetc(pf);  // 得到下一个字符的ASCLL码值,函数会自动往下找
	printf("%c\n", ch);
	ch = fgetc(pf);  // 得到下一个字符的ASCLL码值,函数会自动往下找
	printf("%c\n", ch);
	ch = fgetc(pf);  // 得到下一个字符的ASCLL码值,函数会自动往下找
	printf("%c\n", ch);

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

	return 0;
}

在这里插入图片描述
当然我们也可以不必这么麻烦,直接写个循环瞬间将文件里的字符弄出来打印。

fputc

fputc是将字符写入文件当中,既然是写入,回顾写(w)的特点,如果文件不存在,自动创建一个文件,如果文件存在,则覆盖那个已经存在的文件。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

这里不创建文件,直接使用(当前文件里没有test.txt文件):

#include <stdio.h>
#include <stdlib.h>

int main()
{
	// fputc 将字符写入文件,所以这里用w
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror("open fail");
		exit(-1);
	}

	// 打开文件成功
	// 开始写入
	for (char ch = 'a'; ch <= 'z'; ++ch)
	{
		// 循环写入a-z 26 个字母
		fputc(ch, pf);
	}

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

	return 0;
}

我们打开test.txt文件观察是否写入:

在这里插入图片描述

可以看到,的确将26个字母写入文件。

fgets

fgetsfgetc很相似,只有s和c的差别,但fgets是从文件里得到一行字符(一读就读一行),也就是一行字符串,而fgetc只是得到一个字符,这功能上的差别还是很大的。
在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

下面就以在上面由fputc写好的文件test.txt来作为fgets使用的文件对象来展示fgets的使用效果:

#include <stdio.h>
#include <stdlib.h>

int main()
{
	                // 得字符串 用 'r'
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror("open fail");
		exit(-1);
	}

	// 打开文件成功
	// 开始从文件读字符串
	char buf[100];
	// 从pf中读取十个,用buf指向这一字符串
	fgets(buf, 10, pf);
	printf("%s\n", buf);

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

	return 0;
}

在这里插入图片描述

fputs

理解了fgets函数后,该函数得功能也就很清楚了:将字符串输入到文件中
在这里插入图片描述
在这里插入图片描述

该函数的使用:

#include <stdio.h>
#include <stdlib.h>

int main()
{
	// 输入字符串到文件中 用        'w'
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror("open fail");
		exit(-1);
	}

	// 打开文件成功
	// 输入字符串到文件中
	char s[] = "I love you!";
	fputs(s, pf);

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

	return 0;
}

在这里插入图片描述

fprintf

该函数是以格式化的形式向文件输入数据。
在这里插入图片描述
在这里插入图片描述

fprintf纯概念有些难理解,使用起来其实没那么难:

#include <stdio.h> // 对应头文件
#include <stdlib.h>

struct S
{
	char s[20];
	int a;
	float b;
};

int main()
{
	// 结构体成员数据
	struct S s = { "zhangsan", 520, 13.14 };

	// 输入数据到文件中 用          'w'
	FILE* pf = fopen("test.txt", "w");
	if (pf == NULL)
	{
		perror("open fail");
		exit(-1);
	}

	// 打开文件成功
	// 输入数据到文件中
	fprintf(pf, "%s %d %f", s.s, s.a, s.b);
	// printf("%s %d %f", s.s, s.a, s.b);

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

	return 0;
}

在这里插入图片描述

fscanf

有了fprintf的理解,fscanf也很容易明白了,scanf是从键盘输入到内存当中,而fscanf则是从文件当中读取数据输入到内存当中。
在这里插入图片描述
在这里插入图片描述

我们将上面fprintf输入到文件test.txt的数据在读取出来放入一个结构体当中:
在这里插入图片描述

#include <stdio.h> // 对应头文件
#include <stdlib.h>

struct S
{
	char s[20];
	int a;
	float b;
};

int main()
{
	// 结构体成员数据
	struct S s = { 0 };

	// 输入数据到文件中 用          'r'
	FILE* pf = fopen("test.txt", "r");
	if (pf == NULL)
	{
		perror("open fail");
		exit(-1);
	}

	// 打开文件成功
	// 从文件中读取数据
	fscanf(pf, "%s %d %f", s.s, &s.a, &s.b);
	// 打印结果
	printf("%s %d %f\n", s.s, s.a, s.b);

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

	return 0;
}

在这里插入图片描述

可以看到,fscanf的确把文件中的数据放到了结构体变量s中。

流的介绍

任何一个C语言程序运行的时候,默认打开三个流:

  1. stdin - 标准输入流(键盘)类型:FILE*
  2. stdout - 标准输出流(屏幕)类型:FILE*
  3. stderr - 标准错误流(屏幕)类型:FILE*

以上的函数不但适用于文件,还是用于这三个流,上面的函数图后面一条就表示了,这些函数适用于所有流。

例如:

#include <stdio.h>

int main()
{
	int ch = fgetc(stdin);  // 从键盘输入
	fputc(ch, stdout); // 在屏幕上输出

	return 0;
}

在这里插入图片描述

当然,以上的函数都可以进行相关的操作,流就是用来输入输出的。

fwrite

该函数与上面的函数不同,他是以二进制的方式将数据写入文件,也就是当我们打开写入数据的文件时会发现里面都是一些二进制乱码,完全看不懂。
在这里插入图片描述
在这里插入图片描述

既然是二进制写入,那fopen哪里就要写成wb(表示以二进制写入)

使用如下:

#include <stdio.h>
#include <stdlib.h>

int main()
{
	// fwrite写,二进制形式的写“wb”     
	FILE* pf = fopen("test.txt", "wb");
	if (pf == NULL)
	{
		perror("open fail");
		exit(-1);
	}

	// 打开成功
	int arr[] = { 1,2,3,4,5,6 };

	// 从arr处写入   写入的大小为整个数组的大小   写入1个   写到pf指向的文件当中
	for (int i = 0; i < 6; i++)
	{
	    // 循环将arr中的每一个元素写入文件
		fwrite(arr + i, sizeof(int), 1, pf);
	}

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

	return 0;
}

在这里插入图片描述

可以看到,文件test.txt里面是一些乱东西。

fread

该函数是进行二进制数据文件的读取的,跟fwrite函数一个读一个写,下面是其函数原型:
在这里插入图片描述
在这里插入图片描述

就将fwrite写进test.txt的二进制乱码读取出来看是不是开始的arr数组数据:

#include <stdio.h>
#include <stdlib.h>

int main()
{
	// fread写,二进制形式的读取“wb”     
	FILE* pf = fopen("test.txt", "rb");
	if (pf == NULL)
	{
		perror("open fail");
		exit(-1);
	}

	// 打开成功
	int arr[20];

	// 将文件的数据写入arr   一个整型一个整型的写入   写入1个   从pf指向的文件中读取
	for (int i = 0; i < 6; i++)
	{
		fread(arr + i, sizeof(int), 1, pf);
	}

	//  打印
	for (int i = 0; i < 6; i++)
	{
		printf("%d ", arr[i]);
	}

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

	return 0;
}

在这里插入图片描述

5. 文本文件和二进制文件

  • 根据数据的组织形式,数据文件被称为文本文件或者二进制文件。

  • 数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件。

  • 如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的文件就是文本文件。

  • 一个数据在内存中是怎么存储的呢?
    字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。

    1. 当以二进制形式存储,文件里是一系列的二进制乱码;
    2. 当以ASCLL形式储存,则是以字符形式表达其面值。

6. 文件读取结束的判定

  • 有两个函数可以用作文件读取结束的判定
    1. feof
    2. ferror

被错误使用的feof

  • 牢记:在文件读取过程中,不能用feof函数的返回值直接用来判断文件的是否结束。而是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束。
  1. 文本文件读取是否结束,判断返回值是否为 EOF ( fgetc ),或者 NULL ( fgets )
    例如:
    fgetc 判断是否为EOF.
    fgets 判断返回值是否为 NULL .
  2. 二进制文件的读取结束判断,判断返回值是否小于实际要读的个数。
    例如:
    fread判断返回值是否小于实际要读的个数。

7. 文件缓冲区

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

在这里插入图片描述

写在最后

文件操作实际上是不常用的,大部分公司都有其内部的文件管理系统,所以,对文件操作的困惑不要停留太久。

感谢阅读本小白的博客,错误的地方请严厉指出噢!

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

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

相关文章

4、数据类型

目录 1. 基本类型 &#xff08;1&#xff09;数值类型 &#xff08;2&#xff09;字符类型 &#xff08;3&#xff09;布尔类型 2. 构造类型 3. 空类型 本文主要介绍基本数据类型。 1. 基本类型 基本类型也就是C语言中的基础类型&#xff0c;其中包括整型、字符型、浮点…

我的 git 实战记录

我的 git 实战记录 1. 将本次提交追加到上一次提交 1.1. 情景描述 有些时候&#xff0c;我提交东西太勤快了&#xff0c;导致有部分内容没有提交上去。分为两种情况&#xff1a; 已经 push 远程仓库还没有 push 到远程仓库 1.2. 已经 push到远程仓库 先 git add 提交修改…

Windows免费MQTT服务器搭建及使用

1、EMQ X&#xff08;简称 EMQ) 简介 EMQ的源代码完全开源&#xff0c;支持百万级并发连接&#xff0c;支持完整的MQTT协议&#xff0c;可以简单方便地部署在Linux、Mac OS X或Windows等平台上&#xff0c;支持MySQL、Redis等扩展插件&#xff0c;总的来说&#xff0c;EMQ满足…

Java设计模式-访问者模式Visitor

介绍 访问者模式&#xff08;Visitor Pattern&#xff09;&#xff0c;封装一些作用于某种数据结构的各元素的操作&#xff0c;它可以在不改变数据结构的前 提下定义作用于这些元素的新的操作。主要将数据结构与数据操作分离&#xff0c;解决 数据结构和操作耦合性问题访问者模…

day15|513.找树左下角的值、112. 路径总和、113.路径总和ii、106.从中序与后序遍历序列构造二叉树、105.从前序与中序遍历序列构造二叉树

513.找树左下角的值 给定一个二叉树的 根节点 root&#xff0c;请找出该二叉树的 最底层 最左边 节点的值。 假设二叉树中至少有一个节点。 示例 1: 输入: root [2,1,3] 输出: 1 示例 2: 输入: [1,2,3,4,null,5,6,null,null,7] 输出: 7 问题分析&#xff1a; 递归法&#x…

【基础篇】3 # 数组:为什么很多编程语言中数组都从0开始编号?

说明 【数据结构与算法之美】专栏学习笔记 什么是数组&#xff1f; 数组&#xff08;Array&#xff09;是一种线性表数据结构。它用一组连续的内存空间&#xff0c;来存储一组具有相同类型的数据。 线性表和非线性表 线性表&#xff08;Linear List&#xff09;&#xff1…

第07章 面向对象编程(中级部分OOP)

文章目录IDE(集成开发环境)IDEA介绍idea运行包包的作用包基本语法包的本质分析(原理)包的命名命名规则命名规范常用的包如何引入包注意事项和使用细节访问修饰符【modifier】基本介绍使用的注意事项封装【encapsulation】介绍封装的好处和理解封装的实现步骤封装练习继承【exte…

【阶段四】Python深度学习02篇:深度学习基础知识:神经网络可调超参数:优化器

本篇的思维导图: 神经网络可调超参数:优化器 优化器相当于是用来调解神经网络模型的‘手柄’。 代码 # 编译神经网络,

【C语言】小王带您轻松实现动态内存管理(简单易懂)

在上文通讯录制作中&#xff0c;动态通讯录的使用中就用到了动态内存管理&#xff0c;如果有同学想看一看是如何运用的内存管理函数的&#xff0c;请参考这篇文章&#xff0c;接下来我们一起学习动态内存管理的相关知识。【C语言】使用C语言实现静态、动态的通讯录&#xff08;…

浅显易懂的三次握手与四次挥手

目录 一、三次握手 什么是三次握手&#xff1f; 三次握手图解&#xff1a; 过程解析&#xff1a; &#xff08;1&#xff09;第一次握手&#xff1a; &#xff08;2&#xff09;第二次握手&#xff1a; &#xff08;3&#xff09;第三次握手&#xff1a; 二、四次挥手 …

已解决Python读取20GB超大文件内存溢出报错MemoryError

已解决Python读取20GB超大文件内存溢出报错MemoryError 文章目录报错问题报错翻译报错原因解决方法1解决方法2&#xff08;推荐使用&#xff09;帮忙解决报错问题 日常数据分析工作中&#xff0c;难免碰到数据量特别大的情况&#xff0c;动不动就2、3千万行&#xff0c;如果…

操作系统进程调度算法

进程调度 高级调度&#xff08;作业调度&#xff09;&#xff1a;按一定的原则从外存的作业后备队列中挑选一个作业调入内存&#xff0c;并创建进程。每个作业只调入一次&#xff0c;调出一次。作业调入时会建立PCB&#xff0c;调出时会撤销PCB。 中级调度&#xff08;内存调度…

【历史上的今天】1 月 16 日:互联网工程任务组(IETF)成立;AMD 收购 NexGen;eBay 的第一位员工出生

整理 | 王启隆 透过「历史上的今天」&#xff0c;从过去看未来&#xff0c;从现在亦可以改变未来。 今天是 2023 年 1 月 16 日&#xff0c;在 26 年前的今天&#xff0c;国家电力公司组建成立。电力是运作着我们生活的基本&#xff0c;国家电力公司成立于 1997 年 1 月 16 日…

《深度学习入门基于python的理论与实现》chap2感知机 笔记

《深度学习入门:基于python的理论与实现》chap2 感知机 笔记 3个月前正式开始入坑AI的时候就是看的这本书&#xff0c;当时比较粗略地看到了第六章&#xff0c;没有记笔记&#xff0c;现在来重温一下 文章目录《深度学习入门:基于python的理论与实现》chap2 感知机 笔记2.1 什么…

【阶段四】Python深度学习05篇:深度学习项目实战:卷积神经网络的定义、卷积网络的结构与卷积层的原理

本篇的思维导图: 卷积神经网络的定义 卷积神经网络,简称为卷积网络,与普通神经网络的区别是它的卷积层内的神经元只覆盖输入特征局部范围的单元,具有稀疏连接(sparse connectivity)和权重共享(weight shared)的特点,而且其中的过滤器可以做到对图像关键特征的…

基于Power BI的品牌销售金额帕累托分析

一、原理 帕累托于1906年提出了著名的关于意大利社会财富分配的研究结论&#xff1a;20&#xff05;的人口掌握了80&#xff05;的社会财富。这个结论对大多数国家的社会财富分配情况都成立。因此&#xff0c;该法则又被称为80/20法则。 二、数据源 已知某终端表1《商品信息》…

GO 语言 Web 开发实战一

xdm&#xff0c;咱今天分享一个 golang web 实战的 demo go 的 http 包&#xff0c;以前都有或多或多的提到一些&#xff0c;也有一些笔记在我们的历史文章中&#xff0c;今天来一个简单的实战 HTTP 编程 Get 先来一个 小例子&#xff0c;简单的写一个 Get 请求 拿句柄 设置…

VMware Workstation 17 Pro的下载和安装

目录 一、下载 二、安装 三、检查网络连接 方式一&#xff08;简便版&#xff09; 方式二&#xff08;麻烦版&#xff09; 一、下载 下载地址&#xff1a; Windows 虚拟机 | Workstation Pro | VMware | CN 1、进入该网址后&#xff0c;往下翻&#xff0c;有两个选项&…

并查集是什么?怎么模拟实现?如何应用?

目录 一、什么是并查集&#xff1f; 二、并查集可以解决哪些问题&#xff1f; 三、并查集的模拟实现 3.1、并查集的定义 3.2、查询两个元素是否是同一个集合 3.3、合并两个集合 3.4、求集合个数 3.5、并查集完整代码 小结 一、什么是并查集&#xff1f; 我们可以想象这…

九、MySQL 常用函数汇总(2)

文章目录一、条件判断函数1.1 IF(expr,v1,v2)函数1.2 IFNULL(v1,v2)函数1.3 CASE函数二、系统信息函数2.1 获取MySQL版本号、连接数和数据库名的函数2.2 获取用户名的函数2.3 获取字符串的字符集和排序方式的函数2.4 获取最后一个自动生成的ID值的函数三、加密函数3.1 加密函数…