c语言文件操作-2

上一篇 / 下一篇  2008-07-01 14:22:29 / 个人分类:c语言

数据块读写函数fread和fwrite

  C语言还提供了用于整块数据的读写函数。 可用来读写一组数据,如一个数组元素,一个结构变量的值等。读数据块函数调用的一般形式为: fread(buffer,size,count,fp); 写数据块函数调用的一般形式为: fwrite(buffer,size,count,fp); 其中buffer是一个指针,在fread函数中,它表示存放输入数据的首地址。在fwrite函数中,它表示存放输出数据的首地址。 size 表示数据块的字节数。count 表示要读写的数据块块数。fp 表示文件指针。
例如:
fread(fa,4,5,fp); 其意义是从fp所指的文件中,每次读4个字节(一个实数)送入实数组fa中,连续读5次,即读5个实数到fa中。
从键盘输入两个学生数据,写入一个文件中, 再读出这两个学生的数据显示在屏幕上。
#include<stdio.h>
struct stu
{
char name[10];
int num;
int age;
char addr[15];
}boya[2],boyb[2],*pp,*qq;
main()
{
FILE *fp;
char ch;
int i;
pp=boya;
qq=boyb;
if((fp=fopen("stu_list","wb+"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
printf("\ninput data\n");
for(i=0;i<2;i++,pp++)
scanf("%s%d%d%s",pp->name,&pp->num,&pp->age,pp->addr);
pp=boya;
fwrite(pp,sizeof(struct stu),2,fp);
rewind(fp);
fread(qq,sizeof(struct stu),2,fp);
printf("\n\nname\tnumber age addr\n");
for(i=0;i<2;i++,qq++)
printf("%s\t%5d%7d%s\n",qq->name,qq->num,qq->age,qq->addr);
fclose(fp);
}

  本例程序定义了一个结构stu,说明了两个结构数组boya和 boyb以及两个结构指针变量pp和qq。pp指向boya,qq指向boyb。程序第16行以读写方式打开二进制文件“stu_list”,输入二个学生数据之后,写入该文件中, 然后把文件内部位置指针移到文件首,读出两块学生数据后,在屏幕上显示。

格式化读写函数fscanf和fprintf

fscanf函数,fprintf函数与前面使用的scanf和printf 函数的功能相似,都是格式化读写函数。 两者的区别在于 fscanf 函数和fprintf函数的读写对象不是键盘和显示器,而是磁盘文件。这两个函数的调用格式为: fscanf(文件指针,格式字符串,输入表列); fprintf(文件指针,格式字符串,输出表列); 例如:
fscanf(fp,"%d%s",&i,s);
fprintf(fp,"%d%c",j,ch);
用fscanf和fprintf函数也可以完成例10.6的问题。修改后的程序如例10.7所示。

#include<stdio.h>
struct stu
{
char name[10];
int num;
int age;
char addr[15];
}boya[2],boyb[2],*pp,*qq;
main()
{
FILE *fp;
char ch;
int i;
pp=boya;
qq=boyb;
if((fp=fopen("stu_list","wb+"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
printf("\ninput data\n");
for(i=0;i<2;i++,pp++)
scanf("%s%d%d%s",pp->name,&pp->num,&pp->age,pp->addr);
pp=boya;
for(i=0;i<2;i++,pp++)
fprintf(fp,"%s %d %d %s\n",pp->name,pp->num,pp->age,pp->
addr);
rewind(fp);
for(i=0;i<2;i++,qq++)
fscanf(fp,"%s %d %d %s\n",qq->name,&qq->num,&qq->age,qq->addr);
printf("\n\nname\tnumber age addr\n");
qq=boyb;
for(i=0;i<2;i++,qq++)
printf("%s\t%5d %7d %s\n",qq->name,qq->num, qq->age,
qq->addr);
fclose(fp);
}

文件的随机读写

  前面介绍的对文件的读写方式都是顺序读写, 即读写文件只能从头开始,顺序读写各个数据。 但在实际问题中常要求只读写文件中某一指定的部分。 为了解决这个问题可移动文件内部的位置指针到需要读写的位置,再进行读写,这种读写称为随机读写。 实现随机读写的关键是要按要求移动位置指针,这称为文件的定位。文件定位移动文件内部位置指针的函数主要有两个, 即 rewind 函数和fseek函数。

  rewind函数前面已多次使用过,其调用形式为: rewind(文件指针); 它的功能是把文件内部的位置指针移到文件首。 下面主要介绍
fseek函数。

  fseek函数用来移动文件内部位置指针,其调用形式为: fseek(文件指针,位移量,起始点); 其中:“文件指针”指向被移动的文件。 “位移量”表示移动的字节数,要求位移量是long型数据,以便在文件长度大于64KB 时不会出错。当用常量表示位移量时,要求加后缀“L”。“起始点”表示从何处开始计算位移量,规定的起始点有三种:文件首,当前位置和文件尾。
其表示方法如表10.2。
起始点    表示符号    数字表示
──────────────────────────
文件首    SEEK—SET    0
当前位置   SEEK—CUR    1
文件末尾   SEEK—END     2
例如:
fseek(fp,100L,0);其意义是把位置指针移到离文件首100个字节处。还要说明的是fseek函数一般用于二进制文件。在文本文件中由于要进行转换,故往往计算的位置会出现错误。文件的随机读写在移动位置指针之后, 即可用前面介绍的任一种读写函数进行读写。由于一般是读写一个数据据块,因此常用fread和fwrite函数。下面用例题来说明文件的随机读写。

在学生文件stu list中读出第二个学生的数据。
#include<stdio.h>
struct stu
{
char name[10];
int num;
int age;
char addr[15];
}boy,*qq;
main()
{
FILE *fp;
char ch;
int i=1;
qq=&boy;
if((fp=fopen("stu_list","rb"))==NULL)
{
printf("Cannot open file strike any key exit!");
getch();
exit(1);
}
rewind(fp);
fseek(fp,-i*sizeof(struct stu),2);
fread(qq,sizeof(struct stu),1,fp);
printf("\n\nname\tnumber age addr\n");
printf("%s\t%5d %7d %s\n",qq->name,qq->num,qq->age,
qq->addr);
}

  文件stu_list已由例10.6的程序建立,本程序用随机读出的方法读出第二个学生的数据。程序中定义boy为stu类型变量,qq为指向boy的指针。以读二进制文件方式打开文件,程序第22行移动文件位置指针。其中的i值为1,表示从文件头开始,移动一个stu类型的长度, 然后再读出的数据即为第二个学生的数据。

例8-10] 我们需要同时处理三个文件。文件a d d r. t x t记录了某些人的姓名和地址;文件
t e l . t x t记录了顺序不同的上述人的姓名与电话号码。希望通过对比两个文件,将同一人的姓名、
地址和电话号码记录到第三个文件a d d r t e l . t x t。首先看一下前两个文件的内容:
type addr.txt
h e j i e t i a n j i n g
l i y i n g s h a n g h a i
l i m i n g c h e n g d u
w a n g p i n c h o n g q i n g
type tel.txt?
l i y i n g 1 2 3 4 5
h e j i e 8 7 6 4
w a n g p i n 8 7 6 4 3
l i m i n g 7 6 5 4 3 2 2
这两个文件格式基本一致,姓名字段占1 4个字符,家庭住址或电话号码长度不超过1 4个
字符,并以回车结束。文件结束的最后一行只有回车符,也可以说是长度为0的串。在两个文
件中,由于存放的是同一批人的资料,则文件的记录数是相等的,但存放顺序不同。我们可
以任一文件记录为基准,在另一文件中顺序查找相同姓名的记录,若找到,则合并记录存入第三个文件,将查找文件的指针移到文件头,以备下一次顺序查找。
#include <stdio.h>
#include <stdlib.h>
#include <conio.h>
#include <string.h>
m a i n ( )
{
FILE *fptr1,*fptr2,*fptr3; / * 定义文件指针* /
char temp[15],temp1[15],temp2[15];
if ((fptr1=fopen("addr.txt","r"))==NULL)/ *打开文件* /
{
printf("cannot open file");
e x i t ( 0 ) ;
}
if ((fptr2=fopen("tel.txt","r"))==NULL)
{
printf("cannot open file");
e x i t ( 0 ) ;
}
if ((fptr3=fopen("addrtel.txt","w"))==NULL)
{
printf("cannot open file");
e x i t ( 0 ) ;
}
c l r s c r ( ) ; / *清屏幕* /
while(strlen(fgets(temp1,15,fptr1))>1) 读 /回*的姓名字段长度大于1* /
{
f g e t s ( t e m p 2 , 1 5 , f p t r 1 ) ; / * 读地址* /
f p u t s ( t e m p 1 , f p t r 3 ) ; / * 写入姓名到合并文件* /
f p u t s ( t e m p 2 , f p t r 3 ) ; / * 写入地址到合并文件* /
s t r c p y ( t e m p , t e m p 1 ) ; / * 保存姓名字段* /
do /*查找姓名相同的记录* /
{
f g e t s ( t e m p 1 , 1 5 , f p t r 2 ) ;
f g e t s ( t e m p 2 , 1 5 , f p t r 2 ) ;
} while (strcmp(temp,temp1)!=0);
r e w i n d ( f p t r 2 ) ; / * 将文件指针移到文件头,以备下次查找* /
f p u t s ( t e m p 2 , f p t r 3 ) ; / * 将电话号码写入合并文件* /
}
f c l o s e ( f p t r 1 ) ; / *关闭文件* /
f c l o s e ( f p t r 2 ) ;
f c l o s e ( f p t r 3 ) ;
}


TAG: c语言

 

评分:0

我来说两句

日历

« 2024-05-02  
   1234
567891011
12131415161718
19202122232425
262728293031 

数据统计

  • 访问量: 35857
  • 日志数: 37
  • 文件数: 1
  • 建立时间: 2008-04-03
  • 更新时间: 2008-10-23

RSS订阅

Open Toolbar