怎样向文件读写字符(字符串)
函数名 | 调用形式 | 功能 |
fgetc | fgetc(fp) | 从fp指向的文件读取一个字符 |
fputc | fputc(ch,fp) | 把字符ch写到文件指针变量fp所指向的文件中 |
fgets | fgets(str,n,fp) | 从fp指向的文件中读入一个长度为(n-1)的字符串,存放到字符数组str中,在str末尾添加一'\0' |
fputs | fputs(str,fp) | 把字符数组str写到fp所指向的文件中 |
1、向文件添加数据
#include <stdio.h>
#include <stdlib.h>
int main() { FILE *fp; char ch,filename[20]; printf("请输入所用的文件名:"); scanf("%s",filename); if((fp=fopen(filename,"a"))==NULL) //打开输出文件并使fp指向此文件 { printf("无法打开文件\n"); //如果文件出错,就输出此文件无法打开的信息 exit(0); //终止程序 } ch=getchar(); //用来接收最后输入的回车符 printf("请输入一个将要存储到磁盘的字符串(以#结束):"); ch=getchar(); //接收从键盘输入的第一个字符 while(ch!='#') //当输入'#'时结束循环 { fputc(ch,fp); //向磁盘文件输出一个字符 putchar(ch); //将输出的字符显示在屏幕上 ch=getchar(); //再接收下一个从键盘输入的字符 } fclose(fp); //关闭文件 putchar(10); //向屏幕输出一个换行符 return 0; }
其中:
fp=fopen(filename,"a") 是指向文件添加数据;
fp=fopen(filename,"w")是指向文件写数据;
fp=fopen(filename,"r")是指读取文件的数据。
-----------------------------------------------------------------分割线-------------------------------------------------------------
2、将in指向的文件中的数据复制到out指向的文件
#include <stdio.h> #include <stdlib.h> int main() { FILE *in,*out; //定义指向FILE类型文件的指针变量 char ch,infile[20],outfile[20]; //定义两个字符数组,分别存放两个数据文件名 printf("输入输入文件的名字"); scanf("%s",infile); //输入一个输入文件的名字 printf("输入输出文件的名字"); scanf("%s",outfile); //输入一个输出文件的名字 if((in=fopen(infile,"r"))==NULL) //打开输入文件 { printf("无法打开文件"); exit(0); } if((out=fopen(outfile,"w"))==NULL) //打开输出文件 { printf("无法打开文件"); exit(0); } while(!feof(in)) //如果遇到输入文件的结束标志 { ch=fgetc(in); //从输入文件读入一个字符,暂时放在变量ch中 fputc(ch,out); //将ch写到输出文件中 putchar(ch); //将ch显示在屏幕上 } putchar(10); //完全显示后换行 fclose(in); //关闭文件 fclose(out); return 0; }
1)在访问磁盘文件时,是逐个字符进行的,访问完一个字节后,当前访问位置就指向下一个字符;
2)为了知道对文件的访问是否完成,只需看文件读写位置是否移动到文件的末尾。用feof函数可以检查到文件读写位置标记是否移动到了末尾,既磁盘文件是否结束。如果是函数值为0;
3)运行结果是将将in指向的文件中的数据复制到out指向的文件。前提是in指向的文件必须存在,out指向的文件如果不存在就新建立一个文件。
-----------------------------------------------------------------分割线-------------------------------------------------------------
3、从键盘读入3组字符串,然后将字符串送到指定的磁盘文件中
#include <stdio.h> #include <stdlib.h> #include <string.h> int main() { FILE *fp; char str[3][10],temp[10]; //建立存放字符串的数组 int i,k,j; printf("Enter the string:\n"); for(i=0;i<3;i++) //输入字符串 { gets(str[i]); } if((fp=fopen("F:\\123.txt","w"))==NULL) //打开磁盘文件 { printf("无法打开文件!\n"); exit(0); } printf("\n\n"); for(i=0;i<3;i++) { fputs(str[i],fp); //将字符串str写到fp指向的文件中 fputc(10,fp); //向磁盘输入一个换行符 puts(str[i]); //在屏幕上显示 } fclose(fp); //关闭文件 return 0; }
-----------------------------------------------------------------分割线-------------------------------------------------------------
4、从指定文件中读取字符串并显示出来
#include <stdio.h> #include <stdlib.h> #include <string.h> int main() { FILE *fp; char str[3][10]; //建立存放字符串的数组 int i=0; if((fp=fopen("F:\\123.txt","r"))==NULL) //打开文件 { printf("无法打开文件!"); exit(0); } while(fgets(str[i],10,fp)!=NULL) //从fp指向的文件中读取9个字符到str[i]中 { printf("%s",str[i]); //输出读取的字符串 i++; } fclose(fp); //关闭文件 return 0; }
1)在上一个代码程序中已经将字符串加了换行符,在用fgets函数读取字符串时,指定一次性读取10个字符,但按照fgets函数规定如果遇到"\n"就结束字符串读取,"\n"作为最后一个字符串读到字符数组中;
2)由于读入到字符串数组的每个字符串后面都有一个"\n",所以在向屏幕输出时就不必再加一个"\n",而只要写printf("%s",str[i]); 即可。
-----------------------------------------------------------------分割线-------------------------------------------------------------
5、用格式化的方式读取文件
fprintf(文件指针,格式字符串,输出表列)
例如:fprintf(fp,"%d,%.2f",i,f); 作用是将int型变量 i 和float变量 f 按照%d和%.2f的格式输出到fp指向的文件中。若i=3,f=2.5,
fscanf(文件指针,格式字符串,输入表列)
例如:fscanf(fp,"%d,%.2f",&i,&f); 作用是读取磁盘文件中的int型变量与float型变量分别送给i和f。
用格式化方式对磁盘文件进行读写使用方便,容易理解。但是读写是先要将ASCII码转换成二进制形式再保存 在内存变量中,输出时再转换成ASCII码输出,花费较多时间。
-----------------------------------------------------------------分割线-------------------------------------------------------------
6、用二进制方式向文件读写一组数据
fread(buffer,size,count,fp);
fwrite(buffer,size,count,fp);
buffer是一个地址,是用来存放从文件读入的数据的存储区的起始地址。对于fwrite来说是要把从此地址开始的数据向文件输出;
size要读写的字节数;
count是要读写多少个数据项(一个数据项是size);
fp是指针指向的文件。
例如:
1)fwrite(&stu,4,10,fp);&stu是一个数组的首地址。这个函数将数组stu的10个4字节的数据存储到文件指针fp指向的文件中。
实例:将结构体数组的数据写到文件"F:\\123.txt"中
#include <stdio.h> #include <stdlib.h> #include <string.h> #define SIZE 3 struct Student //建立结构体数组 { char name[10]; int num; int age; char room_number[15]; }stu[SIZE]; void save() //编写一个向文件写数据的函数 { FILE *fp; //文件指针变量 int i; if((fp=fopen("F:\\123.txt","wb"))==NULL) //打开文件 { printf("无法打开文件!"); return; } for(i=0;i<SIZE;i++) { if(fwrite(&stu[i],sizeof(struct Student),1,fp)!=1) //向文件写入数据 printf("文件读写错误!\n"); //错误提醒 } fclose(fp); //关闭文件 } int main() { int i; printf("请输入学生数据:\n"); for(i=0;i<SIZE;i++) //给结构体数组输入数据 scanf("%s%d%d%s",stu[i].name,&stu[i].num,&stu[i].age,stu[i].room_number); save(); return 0; }
2)fread(f,4,10,fp);其中f是一个float型数组名(代表数组首元素地址)。这个函数从fp指向的文件读取10个4字节的数据存储到数组f中。
实例:从文件"F:\\123.txt"中读取数据存储到结构体数组stu中
#include <stdio.h> #include <stdlib.h> #include <string.h> #define SIZE 3 struct Student //建立结构体数组 { char name[10]; int num; int age; char room_number[15]; }stu[SIZE]; int main() { FILE *fp; int i; if((fp=fopen("F:\\123.txt","rb"))==NULL) //打开文件 { printf("无法打开文件!"); exit(0); } for(i=0;i<SIZE;i++) { fread(&stu[i],sizeof(struct Student),1,fp); //从文件中读取数据送到结构体数组中 printf("%-10s %4d %4d %-15s\n",stu[i].name,stu[i].num,stu[i].age,stu[i].room_number); //输出读取的数据 } fclose(fp); //关闭文件 return 0; }
注意:fwrite和fread函数一般用于二进制文件的读写。因为它们是按数据块的长度来决定读写的,不出现字符转换。
-----------------------------------------------------------------分割线-------------------------------------------------------------
7、随机读写文件数据
(1)用rewind函数使文件位置标记指向文件开头
rewind(FILE *)
rewind函数的作用是使文件位置标记重新指向文件开头,此函数没有返回值。
实例:将指定文件的内容显示在屏幕上,然后复制到另一个文件中去
#include <stdio.h> #include <stdlib.h> #include <string.h> int main() { FILE *fp1,*fp2; fp1=fopen("F:\\123.txt","r"); //打开文件 fp2=fopen("F:\\321.txt","w"); //打开文件 while(!feof(fp1)) putchar(getc(fp1)); //从文件中读取数据 putchar(10); //输出一个换行 rewind(fp1); //使fp1重新指向文件的开头 while(!feof(fp1)) putc(getc(fp1),fp2); //将fp1指向的文件中的数据写入fp2中 fclose(fp1); //关闭文件 fclose(fp2); return 0; }
(2)用fseek函数改变文件位置标记
fseek(文件类型指针,位移量,起始点)
“起始点”用0、1、2代替,0表示“文件开始位置”,1表示“当前位置“,2表示”文件末尾位置“。
”位移量“指以”起始点“为起点,向前(后)移动的的字节数。位移量应该是long型数据(在数字的末尾加一个字母L,就表示是long型)。
fseek函数一般用于二进制文件
实例:将数据存入文件中,然后用fseek函数对文件位置标记重新标记,并将移位后的数据输出到屏幕上
#include <stdio.h> #include <stdlib.h> #include <string.h> int main() { FILE *fp; int i; fp=fopen("F:\\123.txt","wb+"); //打开文件 char a[20]="123456789abcdefg"; for(i=0;i<20;i++) // 将数据写入文件中 { fwrite(&a[i],sizeof(a[i]),1,fp); } fseek(fp,3L,0); //重新对文件位置标记进行定位 while(!feof(fp)) { putchar(fgetc(fp)); //将移位后的位置标记后面的数据输出 } fclose(fp); //关闭文件 return 0; }
(3)用ftell函数测定文件位置标记的当前位置
ftell(FILE *)
ftell的作用是得到流式文件中位置标记的当前位置,用相对于文件开头的位移量来表示。如果调用函数出错,ftell函数的返回值为-1L。
实例:将一组数据写入文件中,然后用fseek函数改变文件位置标记并在屏幕上显示当前标记位置
#include <stdio.h> #include <stdlib.h> #include <string.h> int main() { FILE *fp; int i; fp=fopen("F:\\123.txt","wb+"); //打开文件 char a[20]="123456789abcdefg"; for(i=0;i<20;i++) // 将数据写入文件中 { fwrite(&a[i],sizeof(a[i]),1,fp); } fseek(fp,3L,0); //重新对文件位置标记进行定位 i=ftell(fp); //获取文件位置标记的当前位置 if(i==-1L) //出错检测 printf("错误!"); else printf("当前文件位置标记位置:%d\n",i); //输出当前位置值 fclose(fp); //关闭文件 return 0; }
-----------------------------------------------------------------分割线-------------------------------------------------------------
8、文件读写的出错检测
(1)ferror函数
在调用各种文件输入输出函数(fputc、fgetc、fread、fwrite等)时,如果出现错误,可以用ferror函数检查;
ferror(FILE *)
如果ferror函数的返回值是0(假),就表示并未出错,如果返回一个非0值,就表示出错了;
在执行fopen函数时,ferror函数的初始值自动重置为0;
在多次对一个文件调用输入输出函数时,每次调用都会使ferror函数值发生改变,所以,应该在每次调用输入输出函数时都进行一次ferror函数检错,以免信息丢失。
(2)clearerr函数
clearerr函数的作用是使文件错误标志和文件结束标记置为0,从而使文件恢复正常;
clearerr(FILE *)
比如在调用一次输入输出函数时出错,ferror函数的值为一个非0值,应该立即使用clearerr函数使ferror函数值置为0,以便下次检错。