C语言文件操作详解(上)(一)+https://developer.aliyun.com/article/1519328?spm=a2c6h.13148508.setting.14.25634f0e6POlmr
5.fgets()函数的使用
(1)fgets()从文件中读取num-1个字符的情况
下列代码中,arr数组用于接收读到的字符。为了观察更清晰,我们预先在其中放入10个#号。
fgets(arr,5,pf);语句意思是从pf指向的文件中读取5个字符,存入arr中。新存入的字符会覆盖arr中原有的字符。
int main() { //打开文件 FILE* pf = fopen("test.txt", "r"); if (NULL == pf) { perror("fopen"); return 1; } //读文件-一行一行读 //10个# char arr[] = "##########"; fgets(arr, 5, pf); printf("%s", arr); //关闭文件 fclose(pf); pf = NULL; return 0; }
在代码运行前,test.txt文件中的内容如下:
abcdefg\nhijklmnopqrstuvwxyz
运行后,屏幕上显示arr数组内的内容:
如图,虽然我们要输入的num为5,但实际只显示出4个字符。通过调试我们可以发现,是因为系统在读入的时候,自动将在第4个字符后添加了一个'\0':
第5个字符为'\0'
因此,当num小于等于该行字符数时,实际存入arr的是该行前(num-1)个字符,最后一个字符为系统自动追加的'\0'。 而若令要存入的num数大于该行字符总数,则该行字符会被全部读取,包括末尾的'\n'。并且在读完后,还是会在末尾追加'\0'
一行共8个字符,但会另外追加一个'\0'在arr数组中
6.fputs() 文本行输出函数
fputs()函数的API文档:fputs - C++ Reference
函数原型
int fputs ( const char * str, FILE * stream );
C 库函数int fputs(const char *str, FILE *stream)把字符串写入到指定的文件流 stream 中,但不包括空字符。
参数和返回值
- str -- 数组,包含了要写入的以空字符终止的字符序列。
- stream -- 这是指向 FILE 对象的指针,该 FILE 对象标识了要被写入字符串的文件流。
- 返回值 --若成功,返回一个非负值;若发生错误,则返回 EOF。
7.fputs()的使用
打印两行,注意要在字符串中加入换行符'\n',否则并不能实现在文件中显示两行的效果。
int main() { //打开文件 FILE* pf = fopen("test.txt", "w"); if (NULL == pf) { perror("fopen"); return 1; } //写文件-一行一行写 fputs("hello\n", pf); fputs("world!!!!\n", pf); //关闭文件 fclose(pf); pf = NULL; return 0; }
8.fscanf() 格式化输入函数
fscanf()函数的API文档:fscanf - C++ Reference
函数原型
int fscanf ( FILE * stream, const char * format, ... );
C 库函数int fscanf(FILE *stream, const char *format, ...) 从文件流 stream 读取格式化输入。
返回值
- 如果成功,该函数返回成功匹配和赋值的个数。如果到达文件末尾或发生读错误,则返回 EOF。
该函数的用法与scanf函数一致,只是fscanf()函数在参数部分多了一个文件指针用于标识要输入的文件流。
9.fprintf() 格式化输出函数
fprintf()函数的API文档:fprintf - C++ Reference
函数原型
int fprintf ( FILE * stream, const char * format, ... );
该函数的用法与printf函数也一致,只是fprintf函数在参数部分多了一个文件指针用于标识要输出到的文件流。
10.fscanf()与fprintf()函数的使用
(1)读写一个结构体中的数据
先创建一个结构体的数据:
struct S { char name[20]; int age; float score; };
把结构体中的一些数据写到文件中:
int main() { struct S s = { "zhangsan", 20, 95.5f }; //把s中的数据写到文件中 FILE* pf = fopen("test.txt", "w"); if (NULL == pf) { perror("fopen"); return 1; } //写文件 fprintf(pf, "%s %d %.1f", s.name, s.age, s.score); fclose(pf); pf = NULL; return 0; }
运行后效果如下:
2)读取一个结构体的数据
依旧是上述结构体,在写入数据后,我们再将它读取到内存中:
struct S { char name[20]; int age; float score; }; int main() { struct S s = {0}; //把s中的数据写到文件中 FILE* pf = fopen("test.txt", "r"); if (NULL == pf) { perror("fopen"); return 1; } //读文件 fscanf(pf, "%s %d %f", s.name, &(s.age), &(s.score)); printf("%s %d %f\n", s.name, s.age, s.score); fclose(pf); pf = NULL; return 0; }
此时在控制台中也能成功打印出文件内容:
11.fread() 二进制输入
fread()函数的API文档:fread - C++ Reference
函数原型
size_t fread ( void * ptr, size_t size, size_t count, FILE * stream );
从给定文件流stream读取数据到 ptr 所指向的数组中。
参数
- ptr --指向带有最小尺寸 size*nmemb 字节内存块的指针。
- size -- 要读取的每个元素的大小,以字节为单位。
- nmemb-- 元素个数,每个元素的大小为 size 字节。
- stream -- 指向 FILE 对象的指针,该 FILE 对象指定了一个输入文件流。
2.fwrite() 二进制输出
fwrite()函数的API文档:fwrite - C++ Reference
函数原型
size_t fwrite ( const void * ptr, size_t size, size_t count, FILE * stream );
把 ptr 所指向的数组中的数据写入到给定文件流 stream 中。
参数
- ptr -- 这是指向要被写入的元素数组的指针。
- size -- 这是要被写入的每个元素的大小,以字节为单位。
- nmemb -- 这是元素的个数,每个元素的大小为 size 字节。
- nmemb -- 这是指向 FILE 对象的指针,该 FILE 对象指定了一个输出文件流。
13.fread()与fwrite()的使用
(1)二进制的写文件
写一个结构体数据到文件中。注意文件打开模式为"wb"。
struct S { char name[20]; int age; float score; }; int main() { struct S s = { "zhangsan", 20, 95.5f }; //把s中的数据写到文件中 FILE*pf = fopen("test.txt", "wb"); if (NULL == pf) { perror("fopen"); return 1; } //二进制的写文件 fwrite(&s, sizeof(s), 1, pf); fclose(pf); pf = NULL; return 0; }
(2)二进制的读文件
struct S { char name[20]; int age; float score; }; int main() { struct S s = {0}; //把s中的数据写到文件中 FILE* pf = fopen("test.txt", "rb"); if (NULL == pf) { perror("fopen"); return 1; } //二进制的读文件 fread(&s, sizeof(s), 1, pf); printf("%s %d %f\n", s.name, s.age, s.score); fclose(pf); pf = NULL; return 0; }
五、输入输出流的补充
对于任何一个C程序,只要运行起来就默认打开三个流:
stdin -- 标准输入流 -- 键盘
stdout -- 标准输出流 -- 屏幕
stderr -- 标准错误流 -- 屏幕
上面介绍到,除了fread()与fwrite()适用于文件流以外,其余的顺序读写函数适用于所有输入流。因此,它们也可以起到scanf()与printf()的作用,如:
fgetc(stdin); fputc(ch,stdout);
在需要填入一个文件指针的地方,将参数换为stdin或stdout即可。