文章目录
- 一、C文件概述
- 1.基本概念
- 2.文件分类
- 3.C语言对文件的处理方法
- 二、文件的打开与关闭
- 1.文件类型指针
- 2.文件的打开(fopen函数)
- 3.文件的关闭(fclose函数)
- 三、文件的读写
- 1.字符读写函数fgetc和fputc
- 2.字符串读写函数fgets和fputs
- 3.数据块读写函数fread和fwrite
- 4.格式化读写函数fprintf和fscanf
- 5.顺序读写和随机读写
- 四、文件的定位
- 1.fseek函数
- 2.ftell函数
- 五、出错的检测
- 1.ferror函数
- 2.clearerr函数
- 六、文件操作小结
- 1.文件操作
- 2.文件读写
在命令行中封装表白送给她
今天520,是一个表白的好日子,虽然现在还是一条孤独的单身狗,但还是蹭一波热度吧?你也来试一下,给心爱的她一个神秘礼物吧。
一、C文件概述
1.基本概念
文件是指一组相关数据的有序集合,这个数据集有一个名称,叫做文件名。 我们在前面的已经使用到了很多文件,例如源程序文件、目标文件、可执行文件、库文件 (头文件)等。
文件通常是驻留在外部介质(如磁盘等)上的,在使用时才调入内存中来。
操作系统是以文件为单位对数据进行管理的,示意如下:
2.文件分类
从不同的角度可对文件作不同的分类: (1)从用户的角度看,文件可分为:
- 特殊文件(标准输入输出文件或标准设备文件)
- 普通文件(磁盘文件)
(2)从操作系统的角度看,每一个与主机相连的输入、输出设备都看作是一个文件。 例如:
- 输入文件,终端键盘等
- 输出文件,显示屏和打印机等
(3)按数据的组织形式:
- ASCII文件(文本文件) 每一个字节放一个ASCII代码。
- 二进制文件 把内存中的数据按其在内存中的存储形式原样输出到磁盘上存放。
例如整数10000D在内存中的存储形式以及分别按ASCII码形式和二进制形式输出如下图所示:
ASCII文件和二进制文件的比较: ASCII文件便于对字符进行逐个处理,也便于输出字符; 但一般占存储空间较多,而且要花费转换时间。
二进制文件可以节省外存空间和转换时间; 但一个字节并不对应一个字符,不能直接输出字符形式。
一般中间结果数据需要暂时保存在外存上,以后又需要输入内存的,常用二进制文件保存。
3.C语言对文件的处理方法
缓冲文件系统: 系统自动地在内存区为每一个正在使用的文件开辟一个缓冲区。 用缓冲文件系统进行的输入输出又称为高级磁盘输入输出。
非缓冲文件系统: 系统不自动开辟确定大小的缓冲区,而由程序为每个文件设定缓冲区。 用非缓冲文件系统进行的输入输出又称为低级输入输出系统。
在UNIX系统下,用缓冲文件系统来处理文本文件,用非缓冲文件系统来处理二进制文件。
ANSI C标准只采用缓冲文件系统来处理文本文件和二进制文件。
C语言中对文件的读写都是用库函数来实现。
二、文件的打开与关闭
1.文件类型指针
定义文件型指针变量:
代码语言:javascript复制FILE *fp;
fp是一个指向FILE类型结构体的指针变量。
我们使fp指向某一个文件的结构体变量,从而通过该结构体变量中的文件信息能够访问该文件。 如果有n个文件,一般应设n个指针变量,使它们分别指向n个文件,以实现对文件的访问。
定义FILE类型的数组:
代码语言:javascript复制FILE f[5];
定义了一个结构体数组f,它有5个元素,可以用来存放5个文件的信息。
2.文件的打开(fopen函数)
函数调用:
代码语言:javascript复制FILE *fp;
fp = fopen(文件名, 使用文件方式);
参数说明: 文件名是准备访问的文件的名字; 使用文件方式是读还是写等; fp指定指向被打开的文件的指针变量。
文件使用方式如下:
方式 | 含义 |
---|---|
r | (只读)为输入打开一个文本文件 |
w | (只写)为输出打开一个文本文件 |
a | (追加)向文本文件尾增加数据 |
rb | (只读)为输入打开一个二进制文件 |
wb | (只写)为输出打开一个二进制文件 |
ab | (追加)向二进制文件尾增加数据 |
r | (读写)为读/写打开一个文本文件 |
w | (读写)为读/写建立一个新的文本文件 |
a | (读写)为读/写打开一个文本文件 |
rb | |
wb | (读写)为读/写建立一个新的二进制文件 |
ab | (读写)为读/写打开一个二进制文件 |
说明: (1)凡用r方式打开一个文件时,该文件必须已经存在,且只能从该文件读出。
(2)用w打开的文件只能向该文件写入。 若打开的文件不存在,则以指定的文件名建立该文件,若打开的文件已经存在,则将该文件删去,重建一个新文件。
(3)若要向一个已存在的文件追加新的信息,要用a方式打开文件,但此时该文件必须是存在的,否则将会出错。
(4)在打开一个文件时,如果出错,fopen将返回一个空指针值NULL。 在程序中可以用这一信息来判别是否完成打开文件的工作,并作相应的处理。
(5)把一个文本文件读入内存时,要将ASCII码转换成二进制码,而把文件以文本方式写入磁盘时,也要把二进制码转换成ASCII码,因此文本文件的读写要花费较多的转换时间。 对二进制文件的读写不存在这种转换。
文件打开练习如下:
代码语言:javascript复制#include <stdio.h>
#include <stdlib.h>
int main(){
FILE *fp;
if(!(fp = fopen("E:\Test\test.txt", "r"))){
printf("Can not open the file.n");
}
else{
printf("Open success.n");
}
return 0;
}
打印:
代码语言:javascript复制Open success.
3.文件的关闭(fclose函数)
函数调用:
代码语言:javascript复制fclose(文件指针);
函数功能: 使文件指针变量不指向该文件,也就是文件指针变量与文件脱钩,此后不能再通过该指针对原来与其相联系的文件进行读写操作。
返回值: 关闭成功返回值为0,否则返回EOF(-1)。
三、文件的读写
对文件的读和写是最常用的文件操作,在C语言中提供了多种文件读写的函数,使用这些函数时都要包含头文件stdio.h
。
1.字符读写函数fgetc和fputc
fputc()
函数调用:
fputc( ch, fp);
函数功能: 将字符(ch的值)输出到fp所指向的文件中去。
用写或读写方式打开一个已存在的文件时将清除原有的文件内容,写入字符从文件首开始; 如需保留原有文件内容,希望写入的字符以文件末开始存放,必须以追加方式打开文件; 被写入的文件若不存在,则创建该文件。
文件写入练习:
代码语言:javascript复制#include <stdio.h>
#include <stdlib.h>
int main(){
FILE *fp;
char ch, filename[20];
printf("Please input the filename ypu want to write: ");
scanf("%s", filename);
if(!(fp = fopen(filename, "wt "))){
printf("Cannot open the file!n");
exit(0);
}
printf("Please input the sentences you want to write:n");
ch = getchar();
ch = getchar();
while(ch != EOF){
fputc(ch, fp);
ch = getchar();
}
fclose(fp);
return 0;
}
打印:
代码语言:javascript复制Please input the filename ypu want to write: test.txt
Please input the sentences you want to write:
I love c!
^Z
此时查看源程序同级目录,可以看到新增加了一个文件test.txt,里面的内容为:
I love c!
每写入一个字符,文件内部位置指针向后移动一个字节。
fputc函数有一个返回值,如写入成功则返回写入的字符,否则返回一个EOF,可用此来判断写入是否成功。
fgetc()
函数调用:
ch = fgetc(fp);
函数功能: 其意义是从打开的文件fp中读取一个字符并传入ch中。
在fgetc函数调用中,读取的文件必须是以读或读写方式打开的。
在文件内部有一个位置指针,用来指向文件的当前读写字节。
在文件打开时,该指针总是指向文件的第一个字节; 使用fgetc函数后,该位置指针将向后移动一个字节。 因此可连续多次使用fgetc函数,读取多个字符。
文件指针和文件内部的位置指针不是一回事: 文件指针是指向整个文件的,须在程序中定义说明,只要不重新赋值,文件指针的值是不变的; 文件内部的位置指针用以指示文件内部的当前读写位置,每读写一次,该指针均向后移动,它不需在程序中定义说明,而是由系统自动设置的。
文件读出练习:
代码语言:javascript复制#include <stdio.h>
#include <stdlib.h>
int main(){
FILE *fp;
char ch, filename[20];
printf("Please input the filename ypu want to read: ");
scanf("%s", filename);
if(!(fp = fopen(filename, "r"))){
printf("Cannot open the file!n");
exit(0);
}
while(ch != EOF){
ch = fgetc(fp);
putchar(ch);
}
fclose(fp);
return 0;
}
打印:
代码语言:javascript复制Please input the filename ypu want to read: test.txt
I love c!
从一个文本文件顺序读入字符并在屏幕上显示出来:
代码语言:javascript复制ch = fgetc(fp);
while(ch != EOF){
putchar(ch);
ch = fgetc(fp);
}
EOF不是可输出字符,因此不能在屏幕上显示; 由于字符的ASCII码不可能出现-1,因此EOF定义为-1是合适的。 当读入的字符值等于-1时,表示读入的已不是正常的字符而是文件结束符。
从一个二进制文件顺序读入字符:
代码语言:javascript复制while(!feof(fp){
ch = fgetc(fp);
})
ANSI C提供一个feof()
函数来判断文件是否真的结束:
如果是文件结束,函数feof(fp)
的值为1(真);
否则为0(假)。
这也适用于文本文件的读取。
二进制文件读写练习: 实现图片文件合成器。 代码如下:
代码语言:javascript复制#include <stdio.h>
#include <stdlib.h>
int main(){
FILE *f_pic, *f_file, *f_merged;
char pic_name[50], file_name[50], merged_name[50], ch;
printf("请输入要合成的图片和文件的名称:n");
printf("Picture: ");
scanf("%s", pic_name);
printf("File: ");
scanf("%s", file_name);
printf("Result: ");
scanf("%s", merged_name);
if(!(f_pic = fopen(pic_name, "rb"))){
printf("Can not open the picture %s !", pic_name);
return;
}
if(!(f_file = fopen(file_name, "rb"))){
printf("Can not open the file %s !", file_name);
return;
}
if(!(f_merged = fopen(merged_name, "wb"))){
printf("Can not open the file %s !", file_name);
return;
}
while(!(feof(f_pic))){
ch = fgetc(f_pic);
fputc(ch, f_merged);
}
fclose(f_pic);
while(!(feof(f_file))){
ch = fgetc(f_file);
fputc(ch, f_merged);
}
fclose(f_file);
fclose(f_merged);
system("pause");
return 0;
}
打印:
代码语言:javascript复制请输入要合成的图片和文件的名称:
Picture: 520表白.jpg
File: 让人无法拒绝的8句表白.pdf
Result: 520.m
操作示例:
小程序实现封装表白
2.字符串读写函数fgets和fputs
fgets函数调用形式如:
代码语言:javascript复制fgets(str,n,fp);
函数作用:
从fp所指的文件中读出n-1个字符送入字符数组str中,因为在最后加一个'