C语言中的文件操作

由于我们写的程序等数据是存储在电脑的内存上的,如果程序退出,内存回收,就会导致数据丢失,因此我们使用文件就是为了将数据进行持久化的保存。

1.什么是文件

在计算机操作系统中,文件是指存储在硬盘等外存储器上的信息集合。

在程序设计中,我们说的文件有两种:程序文件、数据文件。(这是从文件功能的角度来分类的)

1.1程序文件

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

1.2数据文件

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

1.3文件名

文件名包括三部分:文件路径+文件名主干+文件后缀
如:C:\code\test.txt

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

根据数据的组织形式,数据文件可以分为文本文件和二进制文件

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

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

如:10000在内存中以二进制的形式存储则要在磁盘上占4个字节,而以ASCII码的形式存储在磁盘上则要占5个字节,如图所示。

3.文件的打开与关闭

3.1流和标准流

3.1.1流

我们程序的数据需要输出到各种外部设备,也需要从外部设备中获取数据,不同外部设备的输入和输出操作各不相同。为了方便对各种设备的操作,我们便抽象出流的概念。

我们要想向流中写数据或从流中读取数据,都要打开流,然后再操作。

3.1.2标准流

C语言在启动时,默认打开了3个流:

stdin--标准输入流,一般情况下通过键盘输入,scanf函数就是从标准输入流读取数据。

stdout--标准输出流,一般情况下输出至显示器,printf函数就是将信息输出到标准输出流中。

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

stdin、stdout、stderr这三个流的类型是:FILE*,通常被称为文件指针。

3.2文件指针

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

创建一个指针变量:

FILE* pf;

定义pf是一个指向FILE类型数据的指针变量。可以使pf指向某个文件的文件信息区。通过该文件信息区中的信息就能访问该文件。故,通过文件指针变量能够间接找到与它相关联的文件。

 

3.3文件的打开和关闭

文件在读取之前要先打开文件,而在使用结束后要关闭文件。

ANSIC规定用fopen函数来打开文件,fclose函数来关闭文件。

//打开文件
FILE* pf(const char* filename, const char* mode);
//              文件名                打开方式


//关闭文件
int fclose(FILE* stream);

mode表示文件的打开方式,如下图所示:

 代码示范:

#include <stdio.h>
int main()
{
    //打开文件
    FILE* pf = fopen("data.txt","w");
    if(pf == NULL) //判断文件指针是否是空指针
    {
        perror("fopen");
        return 1;    //如果是则报错并停止
    }
    //文件操作
    //....
    //关闭文件
    fclose(pf);
    pf = NULL;//将pf设为空指针,避免出现野指针
    return 0;
}
    

4.文件的顺序读写

4.1顺序读写函数介绍

上面说的适用于所有输入流一般指适用于标准输入流和其他输入流(如文件输入流);所有输出流也是同理。

根据上面我们可以得出fputc的功能是将一个字符写进流内。其中character是要写的字符,stream是流,返回值为int类型。

如果返回失败,则返回EOF。

代码示例:

#include<stdio.h>
int main()
{
	FILE* pf = fopen("data.txt", "w");
	if (pf == NULL)
	{
		perror("fopen");
		return 0;
	}
	fputc('a', pf);
	fclose(pf);
	pf = NULL;
	return 0;
}

 上面代码运行,则在该代码的路径下创建一个txt文件,上面输出一个字符a,如图:

fgetc的功能是从流中读取一个字符。返回值类型为int,读取失败返回EOF。

代码示范:由于fgetc是从文件中读取数据,故应该在文件中写上内容,如图

int main()
{
	FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen2");
		return 0;
	}
	int ret = fgetc(pf);
	printf("%c\n", ret);
	fclose(pf);
	pf = NULL;
}

 运行结果

 只输出一个a是因为读取数据是根据文件中的光标进行的,运行一次getc则光标往后移一位。

没有运行时,则光标位于上图所示,运行一次fgetc后,光标如下

因此,想要将文件中全部字符输出,可以多运行几次fgetc函数。

在此,就只讲fgetc和fputc,剩下的可以自己了解。

5.文件的随机读写

5.1fseek

文件的随机读写需要用到fseek函数。这个函数是通过文件指针的位置和偏移量来定位文件指针。

其中stream是流,offset是偏移量,origin是起始位置。

上图中,SEEK_SET是文件开头,SEEK_CUR是光标当前位置,SEEK_END是文件结尾。

5.2ftell

该函数的功能是返回文件指针相对于起始位置的偏移量,也就是当前光标位置相对于文件开头的偏移量。

5.3rewind

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

data.txt文件内容为:  

上述三个函数的代码示例:    

int main()
{
	FILE* pf = fopen("data.txt", "r");
	if (pf == NULL)
	{
		perror("fopen");
		return 1;
	}
	fseek(pf, 8, SEEK_SET);   //偏移量为8,起始位置为文件开头,故光标从文件开头向后移8位
	fflush(pf);   //刷新缓冲区,该函数后面会讲到
	int ret = fgetc(pf);   //运行fgetc函数会是光标往后移一位,偏移量变为9
	printf("输出字符为%c\n", ret);
	ret = ftell(pf);   //返回当前光标位置相对于文件开头的偏移量
	printf("偏移量为%d\n", ret);
	rewind(pf);   //将光标移动至文件开头
	ret = fgetc(pf);
	printf("输出字符为%c\n", ret);
	fclose(pf);
	pf = NULL;
	return 0;
}

输出结果:

文件操作中,最重要的就是找到文件中光标的位置,也就是文件指针的位置。

6.文件读取结束的判定

feof

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

ferror的作用:在文件读取结束后,判断是否是因为遇到错误而结束。

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

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

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

7.文件缓冲区

ANSIC标准采用“缓冲文件系统”处理的数据文件,所谓缓冲文件系统是指系统自动的在内存中为程序中每一个正在使用的文件开辟一块“文件缓冲区”。从内存中向磁盘输出数据会先送到内存的缓冲区,将缓冲区装满以后才会一起送到磁盘上。若从磁盘向计算机读入数据,则从磁盘文件中读取数据输入到文件缓冲区,充满缓冲区后,再从缓冲区逐个将数据送到程序数据。

想要将文件缓冲区中的数据及时输出,就要刷新缓冲区或者在文件操作结束后关闭文件。

刷新缓冲区要用函数fflush。

  • 22
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值