【c语言】超详解文件操作

这篇具有很好参考价值的文章主要介绍了【c语言】超详解文件操作。希望对大家有所帮助。如果存在错误或未考虑完全的地方,请大家不吝赐教,您也可以点击"举报违法"按钮提交疑问。

前言:为什么使用文件

       学习过C语言的伙伴,对程序运行结果往往出现在一个黑框里一定不陌生,一旦关闭黑框程序就结束,为数据分配的内存空间也会随之退还。如果你想使用C语言写一个通讯录的话,一定不想黑框关闭存入的数据就丢失,那么此时我们就需要学会使用文件操作。 为了使数据持久的保存在硬盘中,我们可以采用把数据存放在磁盘文件的方式。本篇博客,我们就主要介绍C语言中的文件操作。

一、什么是文件

磁盘上的文件是文件

程序设计中,文件分为两种:程序文件、数据文件(从文件功能的角度来分类的)

1.1程序文件

源文件、目标文件、可执行文件

1.2数据文件

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

1.3文件名

一个文件要有一个唯一的文件标识,以便用户识别和引用。

文件名包含3部分:文件路径+文件名主干+文件后缀

ps:文件名可以没有后缀!!

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

二、文件的打开和关闭

2.1文件指针

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

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

【c语言】超详解文件操作,c语言,开发语言

 结构体中的内容大致如下:

以VS2013编译环境提供的 stdio.h 头文件中的文件类型申明为例子:

【c语言】超详解文件操作,c语言,开发语言

不同的C编译器的FILE类型包含的内容不完全相同。

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

一般都是通过一个FILE的指针来维护这个FILE结构的变量,这样使用起来更加方便。

下面我们可以创建一个FILE*的指针变量:

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

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

2.2文件的打开和关闭

通过文件指针我们就可以操作我们所需要的文件。

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

fopen函数

【c语言】超详解文件操作,c语言,开发语言

该函数有两个参数 一个是你要打开的文件按名称、一个是打开文件的方式

该函数的返回值是一个FILE类型的指针,在文件成功打开后返回此文件信息区的起始地址,打开失败则返回一个NULL(空指针),所以使用fopen时需要一个FILE类型的指针来接收其返回值。

关于文件打开方式的选项列表

【c语言】超详解文件操作,c语言,开发语言

 注:当使用“w”,“wb”,“w+”,“wb+”打开文件时会清除文件原本存储的数据。

fclose函数

【c语言】超详解文件操作,c语言,开发语言stream:传入将要关闭的文件的文件指针。

该函数的返回值是int类型的:如果关闭成功就返回0值,否则返回EOF(-1)值。

示例如下:

#include <stdio.h>
int main()
{
	//打开文件
    FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}

	//...

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

	return 0;
}

三、文件的读写

【c语言】超详解文件操作,c语言,开发语言

3.1顺序读写

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

ps: 粉色标注的后两行 ,这些函数是二进制文件操作,其他函数对文本文件操作

函数的使用 

fputc函数

【c语言】超详解文件操作,c语言,开发语言

代码示例:

#include <stdio.h>
int main()
{
	FILE* pf = fopen("data.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//写文件
	fputc('a', pf);
	fputc('b', pf);
	fputc('c', pf);
	fputc('d', pf);

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

运行结果:

【c语言】超详解文件操作,c语言,开发语言

 解释:这段代码是如何操作的呢?

当第一次打开这个文件,文件内容空白,但有一个标记文件状态的文件指针(注意:这个指针不是pf)是指向文件起始位置,fputc()将a写进去,然后文件指针更新位置,指向下一个位置。每进行一次文件写操作文件指针的位置就发生变化。

fgetc函数

【c语言】超详解文件操作,c语言,开发语言

该函数功能:从文件中读取字符

若读取字符成功,返回一个字符的ASCII码值

若读取失败返回EOF(-1)

代码示例:

#include <stdio.h>
int main()
{
	FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//读文件
	int ch = 0;
	while ((ch = fgetc(pf)) != EOF)
	{
		printf("%c ", ch);
	}
	fclose(pf);
	pf = NULL;
	return 0;
}

运行结果:

 【c语言】超详解文件操作,c语言,开发语言

fgets函数 

 从文件中读取一行

【c语言】超详解文件操作,c语言,开发语言

 可以看到该函数有三个参数:

steam:从哪一个文件里读

str:读完放到哪

num:最多读取的字符个数  (包含‘\0’)

注意:num-1是真正最多读取的字符个数 

若num=100 最多读100-1个字符

代码示例:

#include <stdio.h>
int main()
{
	FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//读文件
	char arr[100] = {0};
	fgets(arr, 100, pf);
	printf("%s\n", arr);
	fgets(arr, 100, pf);
	printf("%s\n", arr);
	fclose(pf);
	pf = NULL;
	return 0;
}

运行结果:

 【c语言】超详解文件操作,c语言,开发语言

注意:虽然规定读取100个字符 ,但遇到\n会自己停止读取当前这一行,第二行虽然还有数据,但不会再向下读取,可以监视arr中的情况,如图所示:

【c语言】超详解文件操作,c语言,开发语言

 

 fputs函数

【c语言】超详解文件操作,c语言,开发语言

代码示例:

#include <stdio.h>
int main()
{
	FILE* pf = fopen("data.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//写文件
	//fputs("hello", pf);
	char arr[] = "hello";
	fputs(arr, pf);
	fputs("world", pf);

	fclose(pf);
	pf = NULL;

	return 0;
}

运行结果:

 【c语言】超详解文件操作,c语言,开发语言

 

 fscanf函数

代码示例;

#include <stdio.h>
struct S
{
	float f;
	char c;
	int n;
};
int main()
{
	struct S s = {0};

	FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//读文件
	fscanf(pf, "%f-%c-%d", &(s.f), &(s.c), &(s.n));
	printf("%f-%c-%d\n", s.f, s.c, s.n);

	fclose(pf);
	pf = NULL;

	return 0;
}

 运行结果:

【c语言】超详解文件操作,c语言,开发语言

fprintf函数

【c语言】超详解文件操作,c语言,开发语言

 代码示例:

struct S
{
	float f;
	char c;
	int n;
};
nt main()
{
	struct S s = { 3.14f, 'w', 100 };

	FILE* pf = fopen("data.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//写文件
	fprintf(pf, "%f-%c-%d", s.f, s.c, s.n);

	fclose(pf);
	pf = NULL;

	return 0;
}

运行结果:

【c语言】超详解文件操作,c语言,开发语言

fread函数
【c语言】超详解文件操作,c语言,开发语言

代码示例:

//二进制的方式读取文件
int main()
{
	int arr[10] = {0};
	//写文件
	FILE* pf = fopen("data.txt", "rb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//二进制的读文件
	fread(arr, sizeof(arr[0]), sizeof(arr) / sizeof(arr[0]), pf);

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

	fclose(pf);
	pf = NULL;

	return 0;
}

运行结果: 

 【c语言】超详解文件操作,c语言,开发语言

fwrite函数
【c语言】超详解文件操作,c语言,开发语言

代码示例:

//二进制的方式写进文件
int main()
{
	int arr[] = { 1,2,3,4,5,6,7,8,9,10 };
	//写文件
	FILE*pf = fopen("data.txt", "wb");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	//二进制的写文件
	fwrite(arr, sizeof(arr[0]), sizeof(arr)/sizeof(arr[0]), pf);

	fclose(pf);
	pf = NULL;

	return 0;
}

运行结果: 

【c语言】超详解文件操作,c语言,开发语言

输入流与输出流

所有输出流 包括:文件流标准输入\输出流

首先流是一个很抽象的概念,可以理解为一个中转站,最终将很多数据传送到外部设备中,外部设备可以是屏幕、硬盘、U盘等等。

C语言只要运行起来,就默认打开了三个流:

1.标准输入流 ------ stdin

2.标准输出流 ------ stdout

3.标准错误流 ------ stderr

注意:stdin、stdout、stderr都属于FILE*类型的指针

在使用fputc函数时,如果我不想将字符写入文件,改为输出到屏幕上,示例代码如下:

#include <stdio.h>
int main()
{
	char ch = 0;
	for (ch = 'a'; ch <= 'z'; ch++)
	{
		if (ch % 5 == 0)
			fputc('\n', stdout);
		fputc(ch, stdout);
	}
	return 0;
}

运行结果: 

【c语言】超详解文件操作,c语言,开发语言

对比一组函数

scanf/fscanf/sscanf

printf/fprintf/sprintf

scanf 是格式化的输入函数,针对是标准输入流(键盘)

printf 是格式化的输出函数,针对的是标准出流(屏幕)

scanf和printf是针对标准输入/输出流的格式化输入/输出函数

fscanf 是针对所有输入流(文件流、标准输入流)的格式化输入函数

fprintf 是针对所有输出流(文件流、标准输出流)的格式化输出函数

sscanf:将字符串转成格式化的数据

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

3.2随机读写

fseek函数

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

int fseek ( FILE * stream, long int offset, int origin );
ftell函数

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

long int ftell ( FILE * stream );
rewind函数

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

void rewind ( FILE * stream );

四、文本文件和二进制文件

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

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

如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。

以ASCII字符的形式存储的文件就是文 本文件。

五、文件读取结束的判定

5.1被错误使用的feof

牢记:在文件读取过程中,不能用feof函数的返回值直接用来判断文件的是否结束。 而是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束。

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

例如:

fgetc 判断是否为 EOF .

fgets 判断返回值是否为 NULL .

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

例如:

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

六、文件缓冲区

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

 

 

#include <stdio.h>
#include <windows.h>
//VS2013 WIN10环境测试
int main()
{
 FILE*pf = fopen("test.txt", "w");
 fputs("abcdef", pf);//先将代码放在输出缓冲区
 printf("睡眠10秒-已经写数据了,打开test.txt文件,发现文件没有内容\n");
 Sleep(10000);
 printf("刷新缓冲区\n");
 fflush(pf);//刷新缓冲区时,才将输出缓冲区的数据写到文件(磁盘)
 //注:fflush 在高版本的VS上不能使用了
 printf("再睡眠10秒-此时,再次打开test.txt文件,文件有内容了\n");
 Sleep(10000);
 fclose(pf);
 //注:fclose在关闭文件的时候,也会刷新缓冲区
 pf = NULL;
 return 0;
}

 这里可以得出一个结论: 因为有缓冲区的存在,C语言在操作文件的时候,需要做刷新缓冲区或者在文件操作结束的时候关闭文 件。 如果不做,可能导致读写文件的问题。文章来源地址https://www.toymoban.com/news/detail-853659.html

到了这里,关于【c语言】超详解文件操作的文章就介绍完了。如果您还想了解更多内容,请在右上角搜索TOY模板网以前的文章或继续浏览下面的相关文章,希望大家以后多多支持TOY模板网!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处: 如若内容造成侵权/违法违规/事实不符,请点击违法举报进行投诉反馈,一经查实,立即删除!

领支付宝红包 赞助服务器费用

相关文章

  • 【c语言】超详解文件操作

           学习过C语言的伙伴,对程序运行结果往往出现在一个黑框里一定不陌生,一旦关闭黑框程序就结束,为数据分配的内存空间也会随之退还。如果你想使用C语言写一个通讯录的话,一定不想黑框关闭存入的数据就丢失,那么此时我们就需要学会使用文件操作。 为了使

    2024年04月16日
    浏览(18)
  • 【c语言】详解文件操作(二)

    fgetc 为字符输入函数, fputc 为字符输出函数,适用所以输入流和输出流 函数原型: 该函数从 stream 指向的输入流中读取 unsigned char 型的下一个字符的值,并将其转换为 int 型,并返回。若在流中 检查到文件末尾 ,则设置该流的文件结束指示符并返回 EOF ;如果 发生读取错误

    2024年02月08日
    浏览(22)
  • 【C语言】文件操作详解(一)

    💐 🌸 🌷 🍀 🌹 🌻 🌺 🍁 🍃 🍂 🌿 🍄🍝 🍛 🍤 📃 个人主页 :阿然成长日记 👈点击可跳转 📆 个人专栏: 🔹数据结构与算法🔹C语言进阶 🚩 不能则学,不知则问,耻于问人,决无长进 🍭 🍯 🍎 🍏 🍊 🍋 🍒 🍇 🍉 🍓 🍑 🍈 🍌 🍐 🍍 为了是数据长久化存

    2024年02月15日
    浏览(25)
  • 【c语言】详解文件操作(一)

    我们都知道程序的 处理结果或计算结果会随着程序的运行结束而消失 ,且如果再次运行程序我们是看不到上次程序的数据的。此时我们就引入了文件的概念,因此 我们将程序运行结束后仍需保存的数值和字符串等数据保存在文件。 那么到底什么是文件呢? 磁盘上的文件是

    2024年02月08日
    浏览(26)
  • C语言 --- 文件操作(万字详解)

    👧个人主页:@小沈熬夜秃头中୧⍤⃝❅ 😚小编介绍:欢迎来到我的乱七八糟小星球🌝 📋专栏:C语言学习 🔑本章内容:文件操作 送给各位💌:“落幕下的花店 玫瑰将荒野的故事告诉风烟 我始终相信 海压竹枝低复举 风吹山角晦还明.” 记得 评论📝 +点赞👍 +收藏😽

    2024年02月13日
    浏览(25)
  • [C语言进阶详解]文件操作(上)

    CSDN的各位友友们你们好,今天千泽为大家带来的是 C语言详解-文件操作篇, 接下来让我们一起了解一下JAVA的学习路线吧! 如果对您有帮助的话希望能够得到您的支持和关注,我会持续更新的! 一.文件的概念 计算机文件,是存储在某种长期储存设备或临时存储设备中的一段数据流

    2023年04月16日
    浏览(30)
  • 【c语言】文件操作 万字详解

            1,程序文件         2,数据文件         3,文件名         1,文件指针         2,文件的打开和关闭         1,顺序读写函数介绍         2,字符输出函数fputc         3,字符输入函数fgetc         4,文本行输出函数fputs         5,文本行输入函

    2024年02月11日
    浏览(40)
  • C语言之文件操作(万字详解)

    个人主页(找往期文章包括但不限于本期文章中不懂的知识点): 我要学编程(ಥ_ಥ)-CSDN博客 目录 前言 文件的打开和关闭  流和标准流 文件指针 文件的打开和关闭  文件的顺序读写  顺序读写函数介绍 fputc的使用  fgetc的使用  fputs的使用  fgets的使用  fprintf的使用  fs

    2024年04月09日
    浏览(21)
  • 【开发语言】C语言与Python的互操作详解

    博主未授权任何人或组织机构转载博主任何原创文章,感谢各位对原创的支持! 博主链接 本人就职于国际知名终端厂商,负责modem芯片研发。 在5G早期负责终端数据业务层、核心网相关的开发工作,目前牵头6G算力网络技术标准研究。 博客内容主要围绕:        5G/6G协议

    2024年02月10日
    浏览(38)
  • 写程序必会的C语言文件操作(上)附手绘图详解

    目录 1. 为什么使用文件  2. 什么是文件 2.1 程序文件  2.2 数据文件 2.3 文件名 3. 文件的打开和关闭 3.1 文件指针 3.2 文件的打开和关闭实例 4. 文件的顺序读写 字符输入函数 fgetc 文本行输出函数 fputs 文本行输入函数 fgets  格式化输出函数 fprintf 格式化输入函数 fscanf sprint

    2023年04月21日
    浏览(36)

觉得文章有用就打赏一下文章作者

支付宝扫一扫打赏

博客赞助

微信扫一扫打赏

请作者喝杯咖啡吧~博客赞助

支付宝扫一扫领取红包,优惠每天领

二维码1

领取红包

二维码2

领红包