您的位置:首页 > 房产 > 建筑 > 【C语言】文件操作(下卷)

【C语言】文件操作(下卷)

2025/1/24 8:26:40 来源:https://blog.csdn.net/2301_82135086/article/details/139434964  浏览:    关键词:【C语言】文件操作(下卷)

前言

在上一卷中,我们知道了文件指针、文件的打开和关闭(打开其他位置的文件)、文件的顺序读写(其中的fputc()、fgetc()),这一卷中,将继续讲解文件操作未讲到的地方。

内容有点多,写不完的部分再开一个文件操作(终卷)。

之前我们讲过的fputc与fgetc是一个一个字符地从文件中读或写入文件,那么我们现在来看看fgets和fputs:

fputs函数

 前一个参数是一个字符指针,指向一个字符串。第二个参数是文件指针,关联到一个文件。返回类型是int。

功能:往流上写一个字符串。(把ptr指向的字符串写到流上。)

使用演示:

在正式写我们的fputs之前,需要的准备有这些:

#include<stdio.h>
int main()
{//1.先得打开文件,才有文件指针FILE* pf = fopen("test.txt", "w");//得判断打开文件是否成功if (pf == NULL){perror("fopen");return 1;}//2.写文件//3。关闭文件fclose(pf);pf = NULL;//避免成为空指针return 0;
}

然后我们就可以写fputs了:

可见运行没有问题,我们就看看这个文件:

这个字符串就被我们成功写入指定文件了。 

那么现在有一个问题,如果我们再写一句,会在两行上还是一行?

 

可以看到, 放在了同一行上。因为我们没有换行。

换行的写法:

在字符串末尾加'\n'就行。

我们再来讲一下返回值int代表什么:

成功的话,非负整数被返回;发生错误,返回EOF并且设置错误指示器。

如果不想接受返回值,就可以像我们上面写的那样使用。

fgets函数

第一个参数是指向字符数组的指针,第二个是整型变量,第三个参数是流,返回值是char*类型的指针。

功能是从流里读一个字符串,读完放到str指向的字符数组中。

num是最多往str中拷贝几个字符(包括用来结束的无效字符)。

使用演示:

现在我们文件里放的是:

然后我们写出这样的代码: 

我们在调试启用后的监视窗口可以观察:

可以看到,我们确实是读了5个字符,但其中最后一个是'\0'。再看打印结果,确实只读了4个有效字符:

 再回去看刚才的函数参数介绍:num是最多往str中拷贝几个字符(包括用来结束的无效字符)。就是这个意思。

在这一句中也有提到:

换行情况

还有一种情况,如果一行的字符数少于我们的num,会换行读吗?

为了演示这种情况,我们先把文件改成这样:

再把代码改成这样(改为读10个字符):

然后看看:

可以看到,我们想读10个字符,却只读到4个有效字符,而在监视窗口中,我们可以看到我们读取到'\n'之后就加上了'\0',读取结束。 

所以fgets在一行字符数充足情况下,读到的有效字符数为num-1;如果一行字符数不够,不会换行读取。  无论是哪种情况,最后都会补上'\0'

所以,fputs fgets的功能是文本行输出/输入函数。

与fputc fgetc相同,这两个函数也是适用于所有输出/输入流,也就是说stdout stdin(屏幕)也适用,那么可以测试一下:

fputs fgets的标准流使用:

注意,返回值可以使用,也可以不接收(不使用)。

fgets读取失败会返回空指针。

fgetc fputc fgets fputs都是针对字符的,但是数据还有整型、浮点型等。

这时我们来看格式化输出、输入函数。fscanf fprintf。可以和scanf printf进行对比。

fprintf函数

 

可以看到二者区别在于第一个参数。fprintf第一个参数是要操作的流,而printf默认操作的流就是stdout。…的参数叫作可变参数列表。在C语言中有这种参数的不多,常见的就是scanf和printf。

比如,下面的例子我们可以看出printf的参数个数确实是可变的:

 而fprintf的使用就是在使用printf的基础上多了第一个参数。演示:

fscanf函数

 

可以看到,和scanf也就差第一个参数。

演示:  

同样的,fprintf fscanf也可以针对标准输出/输入流(屏幕),演示(fprintf针对stdout):

(将printf改为fprintf,前面加上stdout)

 所以我们可以说fprintf fscanf的功能各自包含了printf scanf的功能。

对比一组函数

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

fscanf/ fprintf  针对所有输入/输出流的格式化输入/输出函数(功能更强,可以针对文件和屏幕)

sscanf/sprintf  是干什么的函数?

sprintf函数

可以与printf fprintf进行比较,然后会发现,第二个参数是一样的,而第一个参数不是流,是字符指针。根据其以str为名,其实是指向一个字符数组的指针。

可以在解释中看到其功能是往字符串(数组)中写格式化数据。

注意,因为写到字符数组中去,所以其实是把格式化的数据转换为了字符串。

演示:

所以未来当我们有一些各种类型的数据,想把它们整合为一个字符串时,就可以使用这个函数。 

sscanf函数

这个函数与我们的sprintf就是相反的。 

刚才我们将s中的各个数据转换为了字符串存入了str中,那么现在我们现在从字符串str中提取格式化的数据,还原成一个结构体变量tmp

此时我们可以再与scanf的原型对比,发现就是多出来第一个参数,这个参数是指向一个字符串(数组)的指针。 

此时我们再次观察sprintf sscanf之间的相反性

 

 演示:

  

上面我们说sprintf可以理解为将格式化的数据转换成字符串,那么我们的sscanf也可以理解为将字符串转换成格式化的数据

fwrite函数 

 可以看到这个函数的参数是比较多的。4个参数。

可以看到功能是写数据块到流(文件)中:

将ptr指向空间的内存里count个大小为size字节的元素写到文件中。

比如ptr指向的可能是个数组,数组中放着10个元素,每个元素4个字节,可以一个一个元素写,也可以十个十个元素写。

演示:

 

因为写进文件的是二进制的信息,用文本编辑器是看不懂的。

fread函数

和fwrite的参数是一样的。甚至顺序都没有变化。

和fwrite相反,fread从文件中读取count个大小为size字节的数据,存放在ptr指向的空间中。fwrite是将ptr中count个大小为size字节的数据写到文件中。 

演示:

 这里的88.8变为88.800003是因为浮点数在内存中无法精确保存,会有误差。

(这里还能写成循环。)

fread返回值是什么?

 成功读取几个元素,就返回几。也就是返回实际读取到的个数。

假如8个元素,一次读3个,最后一次只能读到2个。当最后一次返回值比期望的小时明已经是最后一次读取了。

到此为止讲到的这些函数,都是文件的顺序读写。

那么在文件操作(终卷)中,我们再继续看文件的随机读写和其他内容。

到此本文结束,祝阅读愉快^_^

版权声明:

本网仅为发布的内容提供存储空间,不对发表、转载的内容提供任何形式的保证。凡本网注明“来源:XXX网络”的作品,均转载自其它媒体,著作权归作者所有,商业转载请联系作者获得授权,非商业转载请注明出处。

我们尊重并感谢每一位作者,均已注明文章来源和作者。如因作品内容、版权或其它问题,请及时与我们联系,联系邮箱:809451989@qq.com,投稿邮箱:809451989@qq.com