说明结构体变量* /
if ((fp=fopen("test1.txt","w"))==NULL)
{ / *以文本只写方式打开文件* /
printf("cannot open file");
e x i t ( 0 ) ;
}
printf("input data:\\n");
for( i=0;i<2;i )
{
scanf("%s %s %f %f",student.name,student.num,&student.score[0],
&student.score[1]); / *从键盘输入* /
fprintf(fp,"%s %s %7.2f %7.2f\\n",student.name,student.num,
s t u d e n t . s c o r e [ 0 ] , s t u d e n t . s c o r e [ 1 ] ) ; / * 写入文件* /
}
f c l o s e ( f p ) ; / *关闭文件* /
if ((fp=fopen("test.txt","r"))==NULL)
{ /*以文本只读方式重新打开文件* /
printf("cannot open file");
e x i t ( 0 ) ;
}
printf("output from file:\\n");
while (fscanf(fp,"%s %s %f %f\\n",student.name,student.num,
& s t u d e n t . s c o r e [ 0 ] , s t u d e n t . s c o r e [ 1 ] ) ! = E O F )
/ *从文件读入* /
printf("%s %s %7.2f %7.2f\\n",student.name,student.num,student.score[0],student.score[1]); 显/示* 到屏幕*/
fclose(fp); /*关闭文件*/
}
程序设计一个文件变量指针,两次以不同方式打开同一文件,写入和读出格式化数据,有一点很重要,那就是用什么格式写入文件,就一定用什么格式从文件读,否则,读出的数据与格式控制符不一致,就造成数据出错。上述程序运行如下:
此程序所访问的文件也可以定为二进制文件,若打开文件的方式为:
if ((fp=fopen("test1.txt","wb"))==NULL)
{ / * 以二进制只写方式打开文件* /
printf("cannot open file");
e x i t ( 0 ) ;
}
其效果完全相同。
4. 成块读写
前面介绍的几种读写文件的方法,对其复杂的数据类型无法以整体形式向文件写入或从文件读出。C语言提供成块的读写方式来操作文件,使其数组或结构体等类型可以进行一次性读写。成块读写文件函数的调用形式为:
int fread(void *buf,int size,int count,FILE *stream)
int fwrite(void *buf,int size,int count,FILE *stream)
fread ()函数从stream 指向的流文件读取count (字段数)个字段,每个字段为s i z e (字段长度)个字符长,并把它们放到b u f(缓冲区)指向的字符数组中。
fread ()函数返回实际已读取的字段数。若函数调用时要求读取的字段数超过文件存放的字段数,则出错或已到文件尾,实际在操作时应注意检测。
f w r i t e ( )函数从b u f (缓冲区)指向的字符数组中,把c o u n t (字段数)个字段写到s t r e a m所指向的流中,每个字段为s i z e个字符长,函数操作成功时返回所写字段数。
关于成块的文件读写,在创建文件时只能以二进制文件格式创建。
[例8-7] 向磁盘写入格式化数据,再从该文件读出显示到屏幕。
#include "stdio.h"
#include "stdlib.h"
m a i n ( )
{
FILE *fp1;
int i;
struct stu{ / *定义结构体* /
char name[15];
char num[6];
float score[2];
} s t u d e n t ;
if ((fp1=fopen("test.txt","wb"))==NULL)
{ /*以二进制只写方式打开文件* /
printf("cannot open file");
e x i t ( 0 ) ;
}
printf("input data:\\n");
for( i=0;i<2;i ) {
scanf("%s %s %f %f",student.name,student.num,
& s t u d e n t . s c o r e [ 0 ] , & s t u d e n t . s c o r e [ 1 ] ) ; / * 输入一记录* /
fwrite(&student,sizeof(student),1,fp1);成 /块*写入文件*/
}
f c l o s e ( f p 1 ) ;
if ((fp1=fopen("test.txt","rb"))==NULL)
{ /*重新以二进制只写打开文件* /
printf("cannot open file");
e x i t ( 0 ) ;
}
printf("output from file:\\n");
for (i=0;i<2;i )
{
f r e a d ( & s t u d e n t , s i z e o f ( s t u d e n t ) , 1 , f p 1 ) ; / * 从文件成块读* /
printf("%s %s %7.2f %7.2f\\n",student.name,student.num,
s t u d e n t . s c o r e [ 0 ] , s t u d e n t . s c o r e [ 1 ] ) ; / * 显示到屏幕* /
}
f c l o s e ( f p 1 ) ;
}
通常,对于输入数据的格式较为复杂的话,我们可采取将各种格式的数据当做字符串输入,然后将字符串转换为所需的格式。C提供函数:
int atoi(char *ptr)
float atof(char *ptr)
long int atol(char *ptr)
它们分别将字符串转换为整型、实型和长整型。使用时请将其包含的头文件m a t h . h或s t d l i b . h写在程序的前面。
[例8-8] 将输入的不同格式数据以字符串输入,然后将其转换进行文件的成块读写。
#include <stdio.h>
#include <stdlib.h>
m a i n ( )
{
FILE *fp1;
char *temp;
int i;
struct stu{ / *定义结构体类型* /
char name[15]; / *姓名* /
char num[6]; / * 学号* /
float score[2]; / * 二科成绩* /
} s t u d e n t ;
if ((fp1=fopen("test.txt","wb"))==NULL) / * 打开文件* /
{
printf("cannot open file");
e x i t ( 0 ) ;
}
for( i=0;i<2;i ) {
printf("input name:");
g e t s ( s t u d e n t . n a m e ) ; / *输入姓名* /
printf("input num:");
g e t s ( s t u d e n t . n u m ) ; / * 输入学号* /
printf("input score1:");
g e t s ( t e m p ) ; / *输入成绩* /
s t u d e n t . s c o r e [ 0 ] = a t o f ( t e m p ) ;
printf("input score2:");
g e t s ( t e m p ) ;
s t u d e n t . s c o r e [ 1 ] = a t o f ( t e m p ) ;
f w r i t e ( & s t u d e n t , s i z e o f ( s t u d e n t ) , 1 , f p 1 ) ; / *成块写入到文件* /
}
f c l o s e ( f p 1 ) ;
if ((fp1=fopen("test.txt","rb"))==NULL)
{
printf("cannot open file");
e x i t ( 0 ) ;
}
p r i n t f ( " - - - - - - - - - - - - - - - - - - - - - \\ n " ) ;
p r i n t f ( " % - 1 5 s % - 7 s % - 7 s % - 7 s \\ n " , " n a m e " , " n u m " , " s c o r e 1 " , " s c o r e 2 " ) ;
p r i n t f ( " - - - - - - - - - - - - - - - - - - - - - \\ n " ) ;
for (i=0;i<2;i )
{
f r e a d ( & s t u d e n t , s i z e o f ( s t u d e n t ) , 1 , f p 1 ) ;
p r i n t f ( " % - 1 5 s % - 7 s % 7 . 2 f % 7 . 2 f \\ n " , s t u d e n t . n a m e , s t u d e n t . n u m ,
s t u d e n t . s c o r e [ 0 ] , s t u d e n t . s c o r e [ 1 ] ) ;
}
f c l o s e ( f p 1 ) ;
}
}
/ *以下进行文件的随机读写* /
fseek(fp1,3*sizeof(struct stu),0)/;* 定位文件指针指向第三条记录*/
fwrite(&student[1],sizeof(struct stu),1,fp1);
/ * 在第三条记录处写入第二条记录* /
rewind(fp1); /*移动文件指针到文件头* /
p r i n t f ( " - - - - - - - - - - - - - - - - - - - - - \\ n " ) ;
p r i n t f ( " % - 1 5 s % - 7 s % - 7 s % - 7 s \\ n " , " n a m e " , " n u m " , " s c o r e 1 " , " s c o r e 2 " ) ;
p r i n t f ( " - - - - - - - - - - - - - - - - - - - - - \\ n " ) ;
for (i=0;i<n;i )
{ /*重新输出文件内容* /
fread(&student[i],sizeof(struct stu),1,fp1);
p r i n t f ( " % - 1 5 s % - 7 s % 7 . 2 f % 7 . 2 f \\ n " , s t u d e n t [ i ] . n a m e , s t u d e n t [ i ] . n u m ,
s t u d e n t [ i ] . s c o r e [ 0 ] , s t u d e n t [ i ] . s c o r e [ 1 ] ) ;
}
f c l o s e ( f p 1 ) ; / *关闭文件* /
}
程序的第二次输出,即随机访问后,文件中会有两条相同的记录。
e n t , s i z e o f ( s t u d e n t ) , 1 , f p 1 ) ; / *成块写入到文件* /
}
f c l o s e ( f p 1 ) ;
if ((fp1=fopen("test.txt","rb"))==NULL)
{
printf("cannot open file");
e x i t ( 0 ) ;
}
p r i n t f ( " - - - - - - - - - - - - - - - - - - - - - \\ n " ) ;
p r i n t f ( " % - 1 5 s % - 7 s % - 7 s % - 7 s \\ n " , " n a m e " , " n u m " , " s c o r e 1 " , " s c o r e 2 " ) ;
p r i n t f ( " - - - - - - - - - - - - - - - - - - - - - \\ n " ) ;
for (i=0;i<2;i )
{
f r e a d ( & s t u d e n t , s i z e o f ( s t u d e n t ) , 1 , f p 1 ) ;
p r i n t f ( " % - 1 5 s % - 7 s % 7 . 2 f % 7 . 2 f \\ n " , s t u d e n t . n a m e , s t u d e n t . n u m ,
s t u d e n t . s c o r e [ 0 ] , s t u d e n t . s c o r e [ 1 ] ) ;
}
f c l o s e ( f p 1 ) ;
}
8.1.3 随机读写文件
随机对文件的读写是指在文件内部任意对文件内容进行访问,这也就需要对文件进行详
细的定位,只有定位准确,才有可能对文件随机访问。
C语言提供了用于文件定位的函数,它的作用是使文件指针移动到所需要的位置。
int fseek(FILE *fp,long d,int pos)
f p是文件指针,d是位移量,p o s是起始点。
P o s的取值为:
0 :文件开始处
1 :文件的当前位置
2 :文件的尾部
位移量d是l o n g型的数据,可以为正或负值。表示从起始点向下或向上的指针移动。函数
的返回值若操作成功为0,操作失败为非零。
例如:f s e e k ( f p , 5 L , 0 );将文件指针从文件头向下移动5个字节。
f s e e k ( f p , - 1 0 L , 2 );将文件指针从当前位置向上移动1 0个字节。
rewind() 将文件指针移动到文件头。
ftell(FILE *fp) 返回文件指针的当前位置。
[例8-9] 写入5个学生记录,记录内容为学生姓名、学号、两科成绩。写入成功后,随机
读取第三条记录,并用第二条记录替换。
#include <stdio.h>
#include <stdlib.h>
#define n 5
m a i n ( )
{
FILE *fp1; / *定义文件指针* /
char *temp;
int i,j;
struct stu{ / * 定义学生记录结构* /
char name[15];
char num[6];
float score[2];
} s t u d e n t [ n ] ;
if ((fp1=fopen("test.txt","wb"))==NULL) / *以二进制只写方式打开文件* /
{
printf("cannot open file");
e x i t ( 0 ) ;
}
for( i=0;i<n;i )
{
printf("input name:"); / *输入姓名* /
g e t s ( s t u d e n t [ i ] . n a m e ) ;
printf("input num:");
g e t s ( s t u d e n t [ i ] . n u m ) ; / *输入学号* /
printf("input score1:");
g e t s ( t e m p ) ; / *输入一科成绩* /
s t u d e n t [ i ] . s c o r e [ 0 ] = a t o f ( t e m p ) ;
printf("input score2:");
g e t s ( t e m p ) ; / * 输入第二科成绩* /
s t u d e n t [ i ] . s c o r e [ 1 ] = a t o f ( t e m p ) ;
fwrite(&student[i],sizeof(struct stu),1,fp1);成 块/*写入*/
}
fclose(fp1); /*关闭* /
if ((fp1=fopen("test.txt","rb "))==NULL)
{ /*以可读写方式打开文件* /
printf("cannot open file");
e x i t ( 0 ) ;
}
p r i n t f ( " - - - - - - - - - - - - - - - - - - - - - \\ n " ) ;
p r i n t f ( " % - 1 5 s % - 7 s % - 7 s % - 7 s \\ n " , " n a m e " , " n u m " , " s c o r e 1 " , " s c o r e 2 " ) ;
p r i n t f ( " - - - - - - - - - - - - - - - - - - - - - \\ n " ) ;
for (i=0;i<n;i )
{ /*显示全部文件内容* /
fread(&student[i],sizeof(struct stu),1,fp1);
p r i n t f ( " % - 1 5 s % - 7 s % 7 . 2 f % 7 . 2 f \\ n " , s t u d e n t [ i ] . n a m e , s t u d e n t [ i ] . n u m ,
s t u d e n t [ i ] . s c o r e [ 0 ] , s t u d e n t [ i ] . s c o r e [ 1 ] ) ;
}
/ *以下进行文件的随机读写* /
fseek(fp1,3*sizeof(struct stu),0)/;* 定位文件指针指向第三条记录*/
fwrite(&student[1],sizeof(struct stu),1,fp1);
/ * 在第三条记录处写入第二条记录* /
rewind(fp1); /*移动文件指针到文件头* /
p r i n t f ( " - - - - - - - - - - - - - - - - - - - - - \\ n " ) ;
p r i n t f ( " % - 1 5 s % - 7 s % - 7 s % - 7 s \\ n " , " n a m e " , " n u m " , " s c o r e 1 " , " s c o r e 2 " ) ;
p r i n t f ( " - - - - - - - - - - - - - - - - - - - - - \\ n " ) ;
for (i=0;i<n;i )
{ /*重新输出文件内容* /
fread(&student[i],sizeof(struct stu),1,fp1);
p r i n t f ( " % - 1 5 s % - 7 s % 7 . 2 f % 7 . 2 f \\ n " , s t u d e n t [ i ] . n a m e , s t u d e n t [ i ] . n u m ,
s t u d e n t [ i ] . s c o r e [ 0 ] , s t u d e n t [ i ] . s c o r e [ 1 ] ) ;
}
f c l o s e ( f p 1 ) ; / *关闭文件* /
}
程序的第二次输出,即随机访问后,文件中会有两条相同的记录。