C语言文件
文件类型
根据数据的组织形式,数据文件被称为文本文件或二进制文件.
数据在内存中以二进制的形式存储,如果不加转换的输出到外存,就是二进制文件.
如果要求在外存上以ASCII码的形式存储,则需要在存储前转换.以ASCII字符的形式存储的文件就是文本文件.
一个数据在内存中是怎样存储的?
1.字符一律以ASCII形式存储.
2.数值型数据既可以用ASCII形式存储,也可以使用二进制形式存储.
如有整数500000,如果以ASCII码的形式输出到磁盘,则磁盘中占用6个字节(每个字符一个字节),而二进制形式输出,则在磁盘上只占4个字节(int占4个字节)
文件缓冲区
ANSIC标准采用"缓冲文件系统"处理数据文件,所谓缓冲系统是指系统自动地在内存中为程序中每一个正在使用的文件开辟一块"文件缓冲区".从内存向磁盘输出数据会先送到内存中的缓冲区(输出缓冲区),装满缓冲区后才一起送到磁盘上.如果从磁盘向计算机读入数据,则从磁盘文件中读取数据输入到内存缓冲区(输入缓冲区),待到充满缓冲区,再从缓冲区逐个地将数据送到程序数据区(程序变量等).缓冲区的大小根据C编译系统决定的.
文件指针
每个被使用的文件都在内存中开辟了一个相应的文件信息区,用来存放文件的相关信息(如文件的名字,文件状态及文件当前的位置等).这些信息是保存在一个结构体变量中的.该结构体类型是有系统声明的,取名FILE.
如(VS2013编译器)在stdio.h头文件中有以下的文件类型声明:
struct _iobuf {
char *_ptr;
int _cnt;
char *_base;
int _flag;
int _file;
int _charbuf;
int _bufsiz;
char *_tmpfname;
};
typedef struct _iobuf FILE;
不同的C编译器的FILE类型包含的内容不完全相同,但是大同小异
每当打开一个文件时,系统会根据文件的情况自动创建一个FILE结构的变量,并填充其中的信息.
一般都是通过一个FILE的指针来维护这个FILE结构的变量.
FILE* pf;//文件指针变量
定义pf是一个指向FILE类型数据的指针变量.可以使pf指向某个文件的文件信息区(是一个结构体变量).通过该文件信息区中的信息就能够访问该文件.也就是说,通过文件指针变量能够找到与它关联的文件.
FILE* fopen(const char* filename,const char* mode);//打开文件
int fclose(FILE* stream);//关闭文件
打开方式如下:
文件使用方式 | 含义 | 如果指定文件不存在 |
---|---|---|
“r”(只读) | 为了输入数据,打开一个已经存在的文本文件 | 出错 |
“w”(只写–覆盖写–创建写) | 为了输出数据,打开一个文本文件 | 创建一个新的文件 |
“a”(追加) | 向文本文件末尾添加数据 | 出错 |
“rb”(只读) | 为了输入数据,打开一个二进制文件 | 出错 |
“wb”(只写) | 为了输出数据,打开一个二进制文件 | 建立一个新文件 |
“ab”(追加) | 向一个二进制文件尾添加数据 | 出错 |
“r+”(读写) | 为了读和写,打开一个文本文件 | 出错 |
“w+”(读写) | 为了读和写,建立一个新的文件 | 创建一个新文件 |
“a+”(读写) | 打开一个文件,在文件尾进行读写 | 创建一个新文件 |
“rb+”(读写) | 为了读和写打开一个二进制文件 | 出错 |
“wb+”(读写) | 为了读和写,新建一个新的二进制文件 | 创建一个新的文件 |
“ab+”(读写) | 打开一个二进制文件,在文件尾进行读和写 | 创建一个新的文件 |
功能 (头文件均是<stdio.h>) | 函数名 | 适用于 |
---|---|---|
字符输入函数 | fgetc | 所有输入流 |
字符输出函数 | fputc | 所有输出流 |
文本行输入函数 | fgets | 所有输入流 |
文本行输出函数 | fputs | 所有输出流 |
格式化输入函数 | fscanf | 所有输入流 |
格式化输出函数 | fprintf | 所有输出流 |
二进制输入 | fread | 文件 |
二进制输出函数 | fwrite | 文件 |
键盘—标准输入设备—stdin
(标准输入流
)
屏幕—标准输出设备—stdout
(标准输出流
)
这两个是程序默认打开的两个流设备
fputc,fgetc
int fputc(int c,FILE* stream);//把字符c输出到文件指针流中
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{
FILE* pfWrite = fopen("test.txt","w");
if (pfWrite == NULL)
{
printf("%s\n", strerror(errno));
return 0;
}
fputc('c', pfWrite);//把字符'c'输出到文件test.txt中
fputc('h', pfWrite);
fputc('i', pfWrite);
fputc('n', pfWrite);
fputc('a', pfWrite);
fclose(pfWrite);
pfWrite = NULL;
return 0;
}
int fgetc(FILE* stream);//从文件指针流中获取一个字符
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{
FILE* pfRead = fopen("test.txt","r");
if (pfRead == NULL)
{
printf("%s\n", strerror(errno));
return 0;
}
//读文件
printf("%c", fgetc(pfRead));//'c'
printf("%c", fgetc(pfRead));//'h'
printf("%c", fgetc(pfRead));//'i'
printf("%c", fgetc(pfRead));//'n'
printf("%c", fgetc(pfRead));//'a'
/*注意:要是把这几条printf语句改成一条printf("%c%c%c%c%c\n", fgetc(pfRead), fgetc(pfRead), fgetc(pfRead), fgetc(pfRead), fgetc(pfRead));
则会输出anihc,因为这几个fgetc函数是从左往右执行的*/
//关闭文件
fclose(pfRead);
pfRead = NULL;
return 0;
}
fputs,fgets
int fputs(const char* string,FILE* stream);//把字符串写到文件中
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{
char* str = "hello world";
FILE* pfWrite = fopen("test.txt","w");
if (pfWrite == NULL)
{
printf("%s\n", strerror(errno));
return 0;
}
//写文件
fputs(str,pfWrite);//把字符串"hello world"写(输出)到文件中
//关闭文件
fclose(pfWrite);
pfWrite = NULL;
return 0;
}
char* fgets(char* string,int n,FILE* stream);//从文件中读取n-1个字符到string所指向的空间(若是文件中有10个字符,而n=6,则从文件中读取5个字符,)
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{
char str[20] = { 0 };
FILE* pfRead = fopen("test.txt", "r");
if (pfRead == NULL)
{
printf("%s\n", strerror(errno));
return 0;
}
else
{
fgets(str, 20, pfRead);
puts(str);//打印一行字符串,字符串后面隐藏的'\0',在打印时系统自动把'\0'转换成'\n'换行符
}
return 0;
}
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{
FILE* pfWrite = fopen("test.txt","w");
if (pfWrite == NULL)
{
printf("%s\n", strerror(errno));
return 0;
}
//写文件
fputs("hello world\n", pfWrite);
fputs("very good\n", pfWrite);
fclose(pfWrite);
pfWrite = NULL;
return 0;
/*char str1[20] = {0};
char str2[20] = { 0 };
FILE* pfRead = fopen("test.txt", "r");
if (pfRead == NULL)
{
printf("%s\n", strerror(errno));
return 0;
}
else
{
fgets(str1, 20,pfRead);
fgets(str2, 20, pfRead);
printf("%s%s", str1, str2);
}
fclose(pfRead);
pfRead=NULL;
return 0;*/
}
输出:
hello world
very good
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
char str[100] = { 0 };
fgets(str, 100, stdin);//从标准输入流读取(也就是从键盘输入)
fputs(str, stdout);//输出到标准输出流
return 0;
}
fprintf,printf,sprintf,fscanf,scanf,sscanf
int fprintf(FILE* stream,const char* format[,argument]...);//针对所有输出流 格式化输出语句
int printf(const char* format[,argument]...);//针对标准输出流 格式化输出语句
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
struct S
{
int num;
char name[10];
float score;
};
int main()
{
struct S s = { 1, "张三", 91.5f };
FILE* pf = fopen("test.txt", "w");
if (pf == NULL)
{
return 0;
}
//格式化的形式写文件
fprintf(pf, "%d %s %f", s.num, s.name, s.score);
fclose(pf);
pf = NULL;
return 0;
}
int fscanf(FILE* stream,const char* format[,argument]...);//针对所有输入流 格式化输入语句
int scanf(const char* format[,argument]...);//针对标准输入流 格式化输入语句
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
struct S
{
int num;
char name[10];
float score;
};
int main()
{
struct S s = {0};
FILE* pf = fopen("test.txt", "r");
if (pf == NULL)
{
return 0;
}
//格式化的输入数据
fscanf(pf, "%d %s %f", &s.num, s.name, &s.score);
printf("%d %s %f\n", s.num, s.name, s.score);//1 张三 91.500000
fclose(pf);
pf = NULL;
return 0;
}
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
struct S
{
int num;
char name[10];
float score;
};
int main()
{
struct S s = {0};
fscanf(stdin, "%d %s %f", &s.num, s.name, &s.score);
fprintf(stdout, "%d %s %f", s.num, s.name, s.score);
return 0;
}
int sscanf(const char* buffer,const char* format[,argument]...);//从字符串中读取格式化的数据
int sprintf(char* buffer,const char* format[,argument]...);//把格式化的数据输出成(存储到)字符串
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
struct S
{
int num;
char name[10];
float score;
};
int main()
{
struct S s = {2,"张三",89.4f};
struct S tmp = { 0 };
char buf[50] = { 0 };
//把格式化的数据转换成字符串存储到buf中
sprintf(buf, "%d %s %f", s.num, s.name, s.score);
printf("%s\n", buf);//2 张三 89.400002
//从buf中读取格式化的数据到tmp中
sscanf(buf, "%d %s %f", &(tmp.num), tmp.name, &(tmp.score));
printf("%d %s %f", tmp.num, tmp.name, tmp.score);//2 张三 89.400002
return 0;
}
fwrite,fread
size_t fwrite(const void* buffer,size_t size,size_t count,FILE* stream);//写数据到一个文件指针流
//size大小(单位:字节),count个数(一次写几个size字节的数据到文件中)
size_t fread(void* buffer,size_t size,size_t count,FILE* stream);//从文件指针流中读数据
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
int arr[] = {1,2,3,4,5,6,7,8,9};
FILE* pf = fopen("test.txt", "wb");
fwrite(arr, 4,9, pf);//二进制的形式写到文件中
//因为arr数组中的元素是int型,为4个字节,所以size为4,一次写9个4字节的数据到文件中
fclose(pf);//关闭文件
pf = NULL;
return 0;
}
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
#include <string.h>
#include <errno.h>
int main()
{
FILE* pfRead = fopen("test.txt","rb");//以只读方式打开一个二进制文件
int arr[9];
if (pfRead == NULL)//打开文件失败
{
printf("%s\n",strerror(errno));//打印错误码对应的错误信息
return 0;
}
else//打开文件成功
{
int i = 0;
fread(arr,4,9,pfRead);//从文件中一次读取9个4字节大小的数据到arr数组中
for (i = 0; i <9; i++)
printf("%d ", arr[i]);
}
fclose(pfRead);
pfRead=NULL;
return 0;
}
文件的随机读写
fseek
根据文件指针的位置和偏移量来定位文件指针
.
函数原型:
int fseek(FILE* stream,long int offset,int origin);
//offset偏移量,origin文件指针的当前位置
SEEK_CUR
文件指针的当前位置
SEEK_END
文件指针的末尾位置
SEEK_SET
文件指针的起始位置
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
FILE* pf = fopen("test.txt", "r");
if (pf == NULL)
{
return 0;
}
//定位文件指针
fseek(pf, -1, SEEK_END);
int c = fgetc(pf);
printf("%c\n", c);
fclose(pf);
pf = NULL;
return 0;
}
ftell
ftell
函数返回文件指针相对于起始位置的偏移量
函数原型:
long int ftell(FILE* stream);
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
//test.txt:hello worldvery good
FILE* pf = fopen("test.txt", "r");
if (pf == NULL)
{
return 0;
}
//定位文件指针
fseek(pf, 2, SEEK_CUR);
int c = fgetc(pf);
printf("%c %d\n", c, ftell(pf));//l 3
fseek(pf, 3, SEEK_CUR);
int a = fgetc(pf);
printf("%c %d", a, ftell(pf));//w 7
fclose(pf);
pf = NULL;
return 0;
}
rewind
rewind
让文件指针的位置回到
文件的起始位置
函数原型:
void rewind(FILE* stream);
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
int n;
char buf[27] = { 0 };
FILE* pf = fopen("test.txt", "w+");
for (n = 'a'; n <= 'z'; n++)
{
fputc(n, pf);
}
rewind(pf);
fread(buf, 1, 26, pf);
fclose(pf);
pf = NULL;
puts(buf);
return 0;
}
ferror和feof
ferror
函数原型:
int ferror(FILE* stream);
ferror
和feof
函数头文件
#include <stdio.h>
feof
函数原型:
int feof(FILE* stream);//返回非零值表示已到达文件尾
当文件读取结束的时候,可以用ferror函数
来判断文件是否读取失败
结束,可以用feof函数
来判断文件是否遇到文件尾
结束.
1.文本文件
读取是否结束,判断返回值是否为EOF(用fgetc判断)
,或者判断返回值是否为NULL(用fgets判断)
2.二进制文件
的读取结束判断,用fread判断返回值是否小于实际要读的个数
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
int c;
FILE* pf = fopen("test.txt", "r");
if (pf == NULL)
{
return 0;
}
while ((c = fgetc(pf) != EOF))//fgetc当文件读取失败或者是遇到文件尾结束的时候都会返回EOF
{
putchar(c);//#define EOF -1,EOF的值是-1
}
//判断是什么原因文件读取结束
if (ferror(pf))//若文件读取失败返回非零值
{
puts("\nerror when reading");
}
else if (feof(pf))//若遇到文件尾读取结束则返回非零值
{
puts("\nend of file reach");
}
fclose(pf);
pf = NULL;
return 0;
}
#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>
int main()
{
int c;
FILE* pf = fopen("test.txt", "rb");
if (pf == NULL)
{
return 0;
}
while (fread(&c, sizeof(char), 1, pf) >= 1)
{
putchar(c);
}
//判断是什么原因文件读取结束
if (ferror(pf))//若文件读取失败返回非零值
{
puts("\nerror when reading");
}
else if (feof(pf))//若遇到文件尾读取结束则返回非零值
{
puts("\nend of file reach");
}
fclose(pf);
pf = NULL;
return 0;
}