目录
零、写在前面的话
一、文件指针
1、文本文件的创建与关闭
2、文件指针的移动
3、文件缓冲区
4、使用建议
二、文件的读写
1、文本文件的读写
(1)、写入文本文件
(2)、读取文本文件
2、二进制文件的读写
(1)、写入二进制文件
(2)、读取二进制文件
简单记录,方便快速查找。记录的函数不会太全,每个功能只记录我认为最方便的方法。
对文件进行读写操作,首先需要创建文件指针(FILE类型)。FILE 类型的变量存放了文件相关的信息(文件位置、缓冲区大小等),在 stdio.h 中声明。之后进行的所有文件相关操作都需要这个指针,创建函数如下所示:
#include
FILE *fopen(const char * pathname, const char * mode);
pathname:文件路径,可以是绝对路径或相对路径。
mode: 文件的操作模式(可读 or 可写 or 可读写),参数如下表所示。
模式 |
含 义 |
r |
只读 |
w |
只写 |
a |
追加只写 |
r+ |
读写 |
w+ |
读写 |
a+ |
读写 |
其实这些模式很好理解:
1、顾名思义,‘r’是read的缩写,表示读文件;‘w’是write的缩写,表示写文件;‘a’是append的缩写,表示向文件中添加内容。
2、所有带 ‘+’ 的都同时具备读写功能。
3、‘a’与‘w’的区别在于,‘a’是在文件原有内容基础上添加,‘w’是删除原有内容,重新写入。
4、 ‘r’和‘r+’这两种模式使用的前提是文件必须存在;其他四种模式,文件如果不存在会创建同名空白文件。
需要提前知晓的时,打开文件一定要记得关闭。
#include
int fclose(FILE *stream);
文件指针的移动不能靠 fp++, fp-- 等寻常手段;FILE 文件指针内部有专门的位置指针执行当前的读写位置,每次执行读写操作都会更新。
// 将文件指针移动到文件开头
void rewind (FILE *stream);
//返回当前文件指针相对于文件开始位置的字节数
long ftell(FILE *stream);
// 任意的移动文件指针,向前向后都可以
int fseek(FILE *stream, long offset, int whence);
offset:文件指针相对于whence的偏移,为正数时文件指针后移,为负数时前移。
whence:从 SEEK_SET、SEEK_CUR、SEEK_END中取值
SEEK_SET | 文件开头 |
SEEK_CUR | 文件指针当前的位置 |
SEEK_END | 文件末尾 |
其他的日后用到了再补充。
往磁盘写数据的时候会先写在文件缓冲区中,等缓冲区满了以后才会写入磁盘。可以使用fflush 函数主动的触发该操作。
当然调用 fclose 函数,在退出前也会执行 fflush 的操作。
int fflush(FILE *file);
使用指针,为防止出现野指针都应进行非空的判断,文件指针也不例外;
文件指针可能创建失败,需要进行非0的判断;
文件指针打开后一定记得关闭,以便释放资源,特别需要注意return、break、continue、goto等语句,防止逻辑的跳跃跳过了文件指针关闭的执行代码;
文件的读写主要根据文件的类型分为两部分:文本文件与二进制文件。
这两种类型文件的区别在于,文本文件中的内容是基于ascall表示的字符串,而二进制文件表示的是具体的数值,可以二进制、十进制或十六进制表示,转换为文本没有任何意义。
注意读写文件的方法好几种,下面我只介绍我最常用的:fprintf 与 fgets。像 fputc、fputs;fscanf、fgetc 读者自己了解。
int fprintf(FILE *stream, const char *format, ...);
其实用法和我们最熟悉的 printf 一样,只是前面多了个 FILE 指针变量参数;区别在于 printf 将文本输出到终端,fprintf 将文本输出到文件。
举例:
#include
int main()
{
char str[20] = "hello world";
FILE *fp = fopen("test.txt","w");
if ( !fp )
{
printf("file open failed\n");
return -1;
}
fprintf(fp," str : %s\n", str);
fclose(fp);
}
char *fgets(char *buff, int size, FILE *stream);
该函数从文件中按行读取文本,函数返回字符串buff,如果返回空,则表示文件读取结束。
buff : 存放读取的文件
size:读取多少个字符
stream:文件指针
举例:
#include
#include
int main()
{
FILE *fp = fopen("test.txt","r");
char buff[301];
if ( !fp )
{
printf("open file failed.\n");
return -1;
}
while (1)
{
memset(buff, 0, sizeof(buff));
if ( !fgets(buff, 301, fp) ) break;
printf("%s",buff);
}
fclose(fp);
}
二进制文件没有行的概念。
size_t fwrite(const void *ptr, size_t size, size_t nmemb, FILE *stream);
ptr:要写入的数据指针
size:写入数据的单位(byte),默认1字节
nmemb:写入的数据的个数,即多少个字节
stream:文件指针
举例:
#include
#include
int main()
{
FILE *fp = fopen("test.txt","w");
if ( !fp )
{
printf("open file failed.\n");
return -1;
}
unsigned char buff[20];
for (int i = 0; i < 20; i++)
{
buff[i] = 0xff;
}
fwrite(buff, 1, sizeof(buff), fp);
fclose(fp);
}
size_t fread(void *ptr, size_t size, size_t nmemb, FILE *stream);
读取结束时返回 0。
ptr:读取文件存放的位置
size:读取二进制内容的单位(byte),默认为1
nmemb:读取多少个字节
stream:文件指针
举例:
#include
#include
int main()
{
FILE *fp = fopen("test.txt","r");
if ( !fp )
{
printf("open file failed.\n");
return -1;
}
unsigned char buff[20];
while (1)
{
if ( !fread(buff, 1, 20, fp) ) break;
for (int i = 0; i < 20; i++)
printf("%x ", buff[i]);
}
fclose(fp);
}