作者:日出等日落
专栏:C语言
只有流过血的手指,才能弹出世间的绝唱。 ——泰戈尔
目录
1.文件的随机读写
1.1 fseek函数
1.1.1 下面使用fseek函数
1.2 ftell函数
1.3 rewind函数
2 . 文本文件和二进制文件
3 . 文件读取结束的判定
被错误使用的feof:
前面我们讲了文件顺序读写的相关函数与操作,现在,我们来谈论一下怎么样才能随机读写;
功能: 根据文件指针的位置和偏移量来定位文件指针。
- 参数:第一个参数是文件指针,第二个参数是偏移量,第三个参数是起始位置 (如下)
- 头文件:#include
- 第三个参数有三种情况:
- 文件开头
- 文件的当前位置
- 文件的末尾
下面我来为大家解释一下:
首先,我这里是先创建一个text.txt文件,以便后续的操作,文件中存放了abcdef数据;
int main()
{
FILE*pf = fopen("text.txt", "r");
if (pf == NULL)
{
perror("fopen()");
return 1;
}
/*读文件*/
int ch = fgetc(pf);
printf("%c\n", ch);
ch = fgetc(pf);
printf("%c\n", ch);
fclose(pf);
pf = NULL;
return 0;
}
刚开始读文件的时候,fgetc获取的是字符的首地址,也就是第一个字符a,当再次使用fgetc函数的时候,他会自动往后挪动一个位置;
运行结果:
代码示例1:
int main()
{
FILE*pf = fopen("text.txt", "r");
if (pf == NULL)
{
perror("fopen()");
return 1;
}
fseek(pf, 3, SEEK_SET);
int ch = fgetc(pf);
printf("%c\n", ch);//d
fclose(pf);
pf = NULL;
return 0;
}
这里使用 文件开头的参数SEEK_SET,偏移量是3,意思是从文件的开头偏移3个位置获取字符。
运行结果:
代码示例2:
int main()
{
FILE*pf = fopen("text.txt", "r");
if (pf == NULL)
{
perror("fopen()");
return 1;
}
fseek(pf, -3, SEEK_END);
int ch = fgetc(pf);
printf("%c\n", ch);
fclose(pf);
pf = NULL;
return 0;
}
这里使用文件末尾参数SEEK_END,偏移-3个偏移量,意思是从文件的末尾往回偏移3个位置获取字符。
运行结果:
代码示例3:
int main()
{
FILE*pf = fopen("text.txt", "r");
if (pf == NULL)
{
perror("fopen()");
return 1;
}
/*读文件*/
int ch = fgetc(pf);
printf("%c\n", ch);//a
fseek(pf, 2, SEEK_CUR);
ch = fgetc(pf);
printf("%c\n", ch);//d
fclose(pf);
pf = NULL;
return 0;
}
这里使用文件当前位置参数SEEK_CUR,偏移2个偏移量,意思是从文件的当前位置偏移2个位置获取字符。
运行结果:
功能:返回文件指针相对于起始位置的偏移量
代码示例:
int main()
{
FILE*pf = fopen("text.txt", "r");
if (pf == NULL)
{
perror("fopen()");
return 1;
}
/*读文件*/
int ch = fgetc(pf);
printf("%c\n", ch);//a
fseek(pf, 2, SEEK_CUR);
ch = fgetc(pf);
printf("%c\n", ch);//d
int pos = ftell(pf);
printf("%d\n", pos);
fclose(pf);
pf = NULL;
return 0;
}
这里我们刚开始使用fgets获取了一个字符a,那么他就会自动向后移动一个位置,这时的地址就会指向b,然后使用fseek函数从当前位置偏移2个位置,这是输出的是d,但是他在使用过后,它会自动往后移动一个字符,所以,再次调用ftell函数返回文件指针相对于起始位置的偏移量时就是4,就是字符e到起始位置a。
运行结果:
功能:让文件指针的位置回到文件的起始位置
代码示例:
int main()
{
FILE*pf = fopen("text.txt", "r");
if (pf == NULL)
{
perror("fopen()");
return 1;
}
/*读文件*/
int ch = fgetc(pf);
printf("%c\n", ch);//a
fseek(pf, 2, SEEK_CUR);
ch = fgetc(pf);
printf("%c\n", ch);//d
int pos = ftell(pf);
printf("%d\n", pos);
rewind(pf);
ch = fgetc(pf);
printf("%c\n", ch);//a
fclose(pf);
pf = NULL;
return 0;
}
运行结果:
我们接着往下使用rewind函数,他会让文件指针的位置回到文件的起始位置
- 根据数据的组织形式,数据文件被称为文本文件或者二进制文件。
- 数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件。
- 如果要求在外存上以ASCII码的形式存储,则需要在存储前转换。以ASCII字符的形式存储的文件就是文 本文件。
- 一个数据在内存中是怎么存储的呢?
- 字符一律以ASCII形式存储,数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储。 如有整数10000,如果以ASCII码的形式输出到磁盘,则磁盘中占用5个字节(每个字符一个字节),而 二进制形式输出,则在磁盘上只占4个字节(VS2013测试)。
测试用例:
#include
int main()
{
int a = 10000;
FILE* pf = fopen("test.txt", "wb");
fwrite(&a, 4, 1, pf);//二进制的形式写到文件中
fclose(pf);
pf = NULL;
return 0;
}
运行结果:
这就是二进制文件,我们看不懂
牢记:在文件读取过程中,不能用feof函数的返回值直接用来判断文件的是否结束。
而是应用于当文件读取结束的时候,判断是读取失败结束,还是遇到文件尾结束。
1. 文本文件读取是否结束,判断返回值是否为 EOF ( fgetc ),或者 NULL ( fgets )
例如:
fgetc 判断是否为 EOF . fgets 判断返回值是否为 NULL .
2. 二进制文件的读取结束判断,判断返回值是否小于实际要读的个数。
例如:fread判断返回值是否小于实际要读的个数。
- fgetc
如果读取正常,会返回读取到的字符的ASCII值
如果读取失败,返回EOF
- fgets
如果读取正常,返回的是存放读取到的数据的地址
如果读取失败,返回的NULL
- fscanf
如果读取正常,返回的是格式串中指定的数据的个数
如果读取失败,返回的是小于格式串中指定的数据的个数
欢迎大佬们指点!!!!!