前言
在上一卷中,我们知道了文件指针、文件的打开和关闭(打开其他位置的文件)、文件的顺序读写(其中的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个。当最后一次返回值比期望的小时明已经是最后一次读取了。
到此为止讲到的这些函数,都是文件的顺序读写。
那么在文件操作(终卷)中,我们再继续看文件的随机读写和其他内容。
到此本文结束,祝阅读愉快^_^