文件操作大致分三步
- 打开文件
- 读写文件
- 关闭文件
二进制和文本模式的区别
- 在windows系统中,文本模式下,文件以”\r\n”代表换行。若以文本模式打开文件,并用fputs等函数写入换行符”\n”时,函数会自动在”\n”前面加上”\r”。即实际写入文件的是”\r\n” 。
- 在类Unix/Linux系统中文本模式下,文件以”\n”代表换行。所以Linux系统中在文本模式和二进制模式下并无区别
- 对于GBK编码的汉字,一个汉字两个字节,对于utf8来讲一个汉字3个字节,但如果英文字母都是一个字节
fopen
1 | FILE *p = fopen("a.txt", "r"); |
第一个参数指定打开的文件名,第二个参数指定打开方式
返回值如果是NULL,代表打开失败,打开成功返回文件流
关于第二个参数的说明如下:r
:以只读方式打开文件,该文件必须存在。r+
:以可读写方式打开文件,该文件必须存在。用r+写文件时候,从文件开始位置写入rb+
:读写打开一个二进制文件,允许读写数据,文件必须存在。rw+
:读写打开一个文本文件,允许读和写。w
:打开只写文件,若文件存在则文件长度清为0,即该文件内容会消失。若文件不存在则建立该文件。wb
:
按照二进制格式写文件,在windows下不会自动添加'\r'
,wb
:只对windows有效,对于linux而言,w与wb效果相同。w+
:打开可读写文件,若文件存在则文件长度清为零,即该文件内容会消失。若文件不存在则建立该文件。a
:以附加的方式打开只写文件。若文件不存在,则会建立该文件,行为和w
是一样,如果文件存在,写入的数据会被加到文件尾,即文件原先的内容会被保留。(EOF符保留)a+
:以附加方式打开可读写的文件。若文件不存在,则会建立该文件,如果文件存在,写入的数据会被加到文件尾后,即文件原先的内容会被保留。 (原来的EOF符不保留)b
:只对windows有效,对于unix来讲是无效
fclose
fclose关闭fopen打开的文件1
fclose(p);
getc和putc
getc从文件中读取一个字符1
2
3
4
5
6
7
8
9
10
11int main()
{
FILE *fp = fopen("a.txt", "r");
char c;
while ((c = getc(fp)) != EOF)
{
printf("%c", c);
}
fclose(fp);
return 0;
}
putc向文件中写入一个字符1
2
3
4
5
6
7
8
9
10
11
12int main()
{
FILE *fp = fopen("a.txt", "w");
const char *s = "hello world";
int i;
for (i = 0; i < strlen(s); i++)
{
putc(s[i], fp);
}
fclose(fp);
return 0;
}
EOF与feof函数文件结尾
对于一个文件来说,文件的结尾是EOF1
2
3
4
5char c = 0;
while((c = getc(p)) != EOF)
{
printf("c = %c\n", c);
}
feof用作判断文件是否读完,如果已经到文件结尾,返回true1
2
3
4
5
6
7
8
9
10FILE *p1 = fopen("a.txt", "r");
FILE *p2 = fopen("b.txt", "w");
char buf[1024] = { 0 };
while(!feof(p1))
{
fgets(buf, sizeof(buf), p1);//读取一行文件
fputs(buf, p2);
}
fclose(p1);
fclose(p2);
fprintf,fscanf,fgets,fputs
这些函数都是通过FILE *来对文件进行读写printf
,fprintf
和sprintf
有何关系?
三者都是格式化输出函数,不同的是输出的对象不一样printf
向标准输出设备输出1
printf("%s", "test");
sprintf
向字符串输出1
2char buf[1024] = { 0 };
sprintf(buf, "%s", "test");
fprintf
向文件输出,严格输入,不会添加'\n'
1
2FILE *p = fopen("a.txt", "w");
fprintf(p, "%s", "test");
fscanf
从文件读取,如果在一行中遇到空格便结束读取本行读取1
2
3FILE *p = fopen("a.txt", "r");
char buf[1024] = { 0 };
fscanf(p, "%s", buf);//读取到的存入buf
fgets
函数每次读取一行,有三个参数,第一个参数代表临时存储位置,第二个参数代表大小,第三个代表读取位置1
fgets(buf, sizeof(buf), p);
fputs
函数每次写入一行,有两个参数,第一个参数是临时存储位置,第二个是写入位置1
fputs(buf, p);
fscanf
不会读取行尾的'\n'
,fgets
会将行尾的'\n'
读取到buf里面
不论fprintf
还是fputs
都不会自动向行尾添加\n
,需要代码中往buf的行尾写\n
才可以达到换行的目录
stat
头文件包含:#include <sys/stat.h>
函数的第一个参数代表文件名,第二个参数是struct stat结构地址1
2
3struct stst st;
memset(&st, 0, sizeof(st));
stst("a.txt", &st);
得到文件的属性,包括文件建立时间,文件大小等信息
fseek
函数设置文件指针stream的位置。如果执行成功,stream将指向以fromwhere为基准,偏移offset(指针偏移量)个字节的位置,函数返回0。如果执行失败则不改变stream指向的位置,函数返回一个非0值
超出文件末尾位置,还是返回0。往回偏移超出首位置,还是返回0
第一个参数stream为文件指针
第二个参数offset为偏移量,单位:字节,正数表示正向偏移,负数表示负向偏移
第三个参数origin设定从文件的哪里开始偏移,可能取值为:SEEK_CUR
(当前位置)、SEEK_END
(文件结尾)或 SEEK_SET
(文件开头)1
fseek(fp, 3, SEEK_SET);
ftell
函数 ftell 用于得到文件位置指针当前位置相对于文件首的偏移字节数。在随机方式存取文件时,由于文件位置频繁的前后移动,程序不容易确定文件的当前位置1
long len = ftell(fp)
fgetpos和fsetpos
fgetpos和fsetpos是ftell和fseek的加强版
fseek与ftell返回的是long类型,如果文件很大,超过long的范围,那么该函数会有问题,fgetpos与fsetpos函数可以处理更大的文件类型
返回值:成功返回0,否则返回非01
2fpos_t ps = 0;
fgetpos(fp, &ps);
1 | fpos_t ps = 2; |
fflush
只有调用了fclose之后,所有的写入操作才真实的把内容写入文件
所有的C库函数文件读写操作都是针对缓冲区进行的,并不是真实的文件
缓冲区写满以后,写入缓冲区
fflush函数可以将缓冲区中任何未写入的数据写入文件中。1
fflush(p);//实时写入到文件中去
修改配置文件,希望修改实时生效,那么每次修改完成之后我们fflush一次
fread和fwrite
fread和fwrite以二进制形式对文件进行操作,不局限于文本文件1
size_t fread(void *buffer, size_t size, size_t count, FILE *stream);
1 | size_t fwrite(const void* buffer, size_t size, size_t count, FILE* stream); |
返回值:返回实际写入或读取的数据块数目
只要读取到文件最后,没有完整的读取一个数据块出来,fread就返回0
第一个参数代表void *
,写入或者读取的缓冲区
第二个参数是代表写入或读取的时候一个单位的大小
第三个参数是代表写入或读取几个单位
第四个参数是FILE *
1
2char buf[1024] = { "test" };
fwrite(buf, strlen(buf), 1, p);
fread与feof的注意点
fopen读到文件末尾,并不立即返回真,再调用一次才会返回真1
2
3
4while (!feof(p))
{
fread(&buf, 1, sizeof(buf), p);
}
1 | while (fread(&buf, 1, sizeof(buf), p)) |