
《《C语言程序设计与数据结构》第10章 文件读写》由会员分享,可在线阅读,更多相关《《C语言程序设计与数据结构》第10章 文件读写(43页珍藏版)》请在文档大全上搜索。
1、C语言程序设计与数据结构第第10章章 文件读写文件读写C语言程序设计与数据结构教学提示:教学提示:存储在变量和数组(即内存)中的数据存储在变量和数组(即内存)中的数据是临时的,这些数据在程序运行结束后会消失,而是临时的,这些数据在程序运行结束后会消失,而文件可以用来永久地保存大量的数据。如果有些数文件可以用来永久地保存大量的数据。如果有些数据需要反复使用或永久保存,应该考虑使用文件来据需要反复使用或永久保存,应该考虑使用文件来完成。完成。教学要求:教学要求:本章要掌握本章要掌握文件在读写之前必须打开,文件在读写之前必须打开,读写结束必须关闭。文件可按只读、只写、读写、读写结束必须关闭。文件可按
2、只读、只写、读写、追加四种操作方式打开,同时还必须指定文件的类追加四种操作方式打开,同时还必须指定文件的类型是二进制文件还是文本文件。文件可按字节,字型是二进制文件还是文本文件。文件可按字节,字符串,数据块为单位读写,文件也可按指定的格式符串,数据块为单位读写,文件也可按指定的格式进行读写。文件内部的位置指针可指示当前的读写进行读写。文件内部的位置指针可指示当前的读写位置,移动该指针可以对文件实现随机读写。位置,移动该指针可以对文件实现随机读写。 C语言程序设计与数据结构10.1文件概述与文件指针 所谓所谓“文件文件”是指一组相关数据的有序集合。是指一组相关数据的有序集合。 我们在前几章中已多
3、次接触使用过文件,例如我们在前几章中已多次接触使用过文件,例如源程序文件、目标文件、可执行文件、库文件源程序文件、目标文件、可执行文件、库文件 (头头文件文件)等。等。 C语言程序设计与数据结构10.1.1 文件分类文件通常是驻留在外部介质文件通常是驻留在外部介质(如磁盘等如磁盘等)上的,在使上的,在使用时才调入内存中来。用时才调入内存中来。C语言把文件看作是由一个语言把文件看作是由一个个字符(字节)的数据顺序组成的。它把数据看作个字符(字节)的数据顺序组成的。它把数据看作是连续的字符(字节)流,这样它对文件的存取实是连续的字符(字节)流,这样它对文件的存取实际上是以字符(字节)为单位的。输入
4、输出的数据际上是以字符(字节)为单位的。输入输出的数据流的开始和结束只受程序的控制而不受特定符号流的开始和结束只受程序的控制而不受特定符号(比如回车换行符)的控制。也就是说,(比如回车换行符)的控制。也就是说,C语言在语言在输出数据到文件中时,并不会自动增加回车换行符输出数据到文件中时,并不会自动增加回车换行符以示结束,在输入时也不会以读入回车换行符作为以示结束,在输入时也不会以读入回车换行符作为数据的间隔。数据的间隔。C语言程序设计与数据结构10.1.1 文件分类文件分类从用户的角度看,文件可分为普通文件和设备文件从用户的角度看,文件可分为普通文件和设备文件两种。两种。根据文件的存储形式,可
5、分为根据文件的存储形式,可分为ASCII码文件和二进码文件和二进制文件两种:制文件两种: (1) ASCII文件也称为文本文件,这种文件在磁盘中文件也称为文本文件,这种文件在磁盘中存放时每个字符对应一个字节,用于存放对应的存放时每个字符对应一个字节,用于存放对应的ASCII码。例如一个整型数据码。例如一个整型数据1234在文本文件中占在文本文件中占4个字节。我们用个字节。我们用Windows自带的记事本软件打开自带的记事本软件打开该类型文件能读懂文件中的内容。该类型文件能读懂文件中的内容。 (2) (2) 二进制文件是按二进制的编码方式来存放文件二进制文件是按二进制的编码方式来存放文件的。例如
6、一个整型数据的。例如一个整型数据230在二进制文件中占在二进制文件中占2个字个字节。节。C语言程序设计与数据结构10.1.2 文件类型指针 在语言中要使用文件,必须用一个指针变量指向一个在语言中要使用文件,必须用一个指针变量指向一个文件,这个指针称为文件指针。通过文件指针就可对它所文件,这个指针称为文件指针。通过文件指针就可对它所指的文件进行各种操作。指的文件进行各种操作。 定义文件指针的一般形式为:定义文件指针的一般形式为:FILE *指针变量标识符;指针变量标识符; 其中其中FILE必须为大写,它是在头文件必须为大写,它是在头文件stdio.h中定义的一中定义的一个结构体,该结构体中含有文
7、件名、文件状态和文件当前个结构体,该结构体中含有文件名、文件状态和文件当前位置等信息,我们编程时不必关心位置等信息,我们编程时不必关心FILE结构的细节。结构的细节。 例如:例如: FILE *fp;表示定义了一个名为表示定义了一个名为fp的文件指针。的文件指针。如果变量如果变量fp已被正确赋值,则可以根据结构变量已被正确赋值,则可以根据结构变量fp所提供的所提供的信息找到一个文件并实施对该文件的操作。习惯上把信息找到一个文件并实施对该文件的操作。习惯上把fp称为称为指向一个文件的指针。指向一个文件的指针。 C语言程序设计与数据结构10.2文件的打开与关闭 使用文件必须遵循使用文件必须遵循“先
8、打开,再对文件进行读先打开,再对文件进行读写操作,最后关闭文件写操作,最后关闭文件”的原则。的原则。 在语言中,对文件的这些操作都可以由库函在语言中,对文件的这些操作都可以由库函数来完成,对用到的库函数的声明包含在文件数来完成,对用到的库函数的声明包含在文件stdio.h中,因此在编程中要操作文件时,要使用头中,因此在编程中要操作文件时,要使用头文件包含命令文件包含命令#include “stdio.h”。 C语言程序设计与数据结构10.2.1文件的打开 所谓打开文件,实际上是获得文件的各种有关所谓打开文件,实际上是获得文件的各种有关信息,并使文件指针指向该文件,以便进行下一步信息,并使文件指
9、针指向该文件,以便进行下一步操作。操作。C语言提供了函数语言提供了函数fopen来打开文件。来打开文件。 其使用格式为:其使用格式为: 文件指针名文件指针名=fopen(文件名文件名, 使用文件方式使用文件方式); 其中,其中,“文件指针名文件指针名”必须是已被说明为必须是已被说明为FILE 类类型的指针变量;型的指针变量;“文件名文件名”是将被打开的文件的文是将被打开的文件的文件名,通常是字符串常量或字符串数组(注意路径件名,通常是字符串常量或字符串数组(注意路径的分隔符应使用的分隔符应使用“”););“使用文件方式使用文件方式”是指是指文件的类型和操作要求,是一个由一对双引号括起文件的类型
10、和操作要求,是一个由一对双引号括起来的字符串,共有来的字符串,共有12种种,具体的符号表示和含义见具体的符号表示和含义见表表10-1。 C语言程序设计与数据结构文件使用方式文件使用方式含义含义“rt”打开一个已经存在的文本文件,只能读取数据“wt”打开一个文本文件,只能写入数据。若文件不存在,则自动建立一个新文件接受写入的数据;若文件存在,则删除文件中原有内容,并接受写入的数据(覆盖)“a”打开一个已经存在的文本文件,只能写入数据并且追加在文件的尾部“r+”打开一个已经存在的文本文件,可以读取数据,也可以写入数据“w+”打开一个文本文件,可以读取数据,也可以写入数据。若文件不存在,则自动建立一
11、个新文件接受写入的数据;若文件存在,则删除文件中原有内容,并接受写入的数据(覆盖)“a+”打开一个已经存在的文本文件,可以读取数据,也可以写入数据(追加在文件的尾部)“rb”打开一个已经存在的二进制文件,只能读取数据“wb”打开一个二进制文件,只能写入数据。若文件不存在,则自动建立一个新文件接受写入的数据;若文件存在,则删除文件中原有内容并接受写入的数据(覆盖)“ab”打开一个已经存在的二进制文件,只能写入数据(追加在文件的尾部)“rb+”打开一个已经存在的二进制文件,可以读取数据,也可以写入数据“wb+”打开一个二进制文件,可以读取数据,也可以写入数据。若文件不存在,则自动建立一个新文件接受
12、写入的数据;若文件存在,则删除文件中原有内容,并接受写入的数据(覆盖)“ab+”打开一个已经存在的二进制文件,可以读取数据,也可以写入数据(追加在文件的尾部)C语言程序设计与数据结构fopen打开文件 例如:例如: FILE *fpTa; fpTa=fopen(c:cprogramabc,rb); /*反斜线反斜线“”中的第一个表示转义字符中的第一个表示转义字符*/ 其意义是打开其意义是打开C驱动器磁盘的驱动器磁盘的cprogram子目录下的子目录下的文件文件abc,这是一个二进制文件,只允许按二进制这是一个二进制文件,只允许按二进制方式进行读操作。方式进行读操作。又如:又如: FILE *f
13、p; fp= fopen (mytest1.txt,r); 其意义是在当前目录下打开文件其意义是在当前目录下打开文件mytest1.txt,只允只允许进行许进行“读读”操作,并操作,并使使fp指向该文件。指向该文件。 C语言程序设计与数据结构10.2.2文件关闭函数 在使用完一个文件以后,应该将该文件关闭,在使用完一个文件以后,应该将该文件关闭,以防止它再被调用或丢失数据。以防止它再被调用或丢失数据。“关闭关闭”就是使文就是使文件指针变量不再指向该文件,也就是文件指针变量件指针变量不再指向该文件,也就是文件指针变量与该文件与该文件“断开断开”,此后不能通过该指针对原来与,此后不能通过该指针对原
14、来与其相联系的文件进行读、写操作(除非再次打开该其相联系的文件进行读、写操作(除非再次打开该文件,使文件指针变量重新指向该文件)。文件,使文件指针变量重新指向该文件)。 fclose()函数用来关闭一个已经由函数用来关闭一个已经由fopen()函数打开函数打开的文件,正常完成关闭文件操作时,的文件,正常完成关闭文件操作时,fclose函数返函数返回值为回值为0,有错误发生则返回非零值。,有错误发生则返回非零值。其调用的一般形式为:其调用的一般形式为: fclose(文件指针文件指针); 例如:例如: fclose(fp); C语言程序设计与数据结构10.3 文件位置指针的有关函数 在文件内部有
15、一个位置指针,用来指向文件的在文件内部有一个位置指针,用来指向文件的当前读写字节。在文件打开时,该位置指针总是指当前读写字节。在文件打开时,该位置指针总是指向文件的第一个字节。使用下面章节介绍的向文件的第一个字节。使用下面章节介绍的fgetc等等函数对文件进行读写后,该位置指针将自动向后移函数对文件进行读写后,该位置指针将自动向后移动。动。 但文件指针和文件内部的位置指针不是一回事。但文件指针和文件内部的位置指针不是一回事。文件指针是指向整个文件的,须在程序中定义说明,文件指针是指向整个文件的,须在程序中定义说明,只要不重新赋值,文件指针的值是不变的。文件内只要不重新赋值,文件指针的值是不变的
16、。文件内部的位置指针用以指示文件内部的当前读写位置,部的位置指针用以指示文件内部的当前读写位置,每读写一次,该指针均自动向后移动,它不需在程每读写一次,该指针均自动向后移动,它不需在程序中定义说明,而是由系统自动设置的。序中定义说明,而是由系统自动设置的。 C语言程序设计与数据结构顺序读写和随机读写文件的读写方式分为顺序读写和随机读写。文件的读写方式分为顺序读写和随机读写。(1) (1) 顺序读写即读写文件只能从头开始,顺序读写各个数据。顺序读写即读写文件只能从头开始,顺序读写各个数据。 但在实际问题中常要求只读写文件中某一指定的部分,为但在实际问题中常要求只读写文件中某一指定的部分,为了解决
17、这个问题可移动文件内部的位置指针到需要读写的了解决这个问题可移动文件内部的位置指针到需要读写的位置,再进行读写,这种读写称为随机读写。位置,再进行读写,这种读写称为随机读写。 (2) (2) 随机读写可以通过利用系统函数去主动移动文件内部随机读写可以通过利用系统函数去主动移动文件内部的位置指针来实现。这样的函数主要有两个,即的位置指针来实现。这样的函数主要有两个,即 rewind 函函数和数和fseek函数。实现随机读写的关键是要按要求移动位置函数。实现随机读写的关键是要按要求移动位置指针,这称为文件的定位。指针,这称为文件的定位。在移动位置指针之后,即可用在移动位置指针之后,即可用后面介绍的
18、任一种读写函数进行读写。由于一般是读写一后面介绍的任一种读写函数进行读写。由于一般是读写一个数据据块,因此常用个数据据块,因此常用fread和和fwrite函数。函数。 C语言程序设计与数据结构rewind函数rewind函数的功能是把文件内部的位置指针移到函数的功能是把文件内部的位置指针移到文件首。其调用形式为:文件首。其调用形式为: rewind(文件指针文件指针);“文件指针文件指针”指向被移动的文件。指向被移动的文件。 C语言程序设计与数据结构fseek函数fseek函数用来移动文件内部位置指针,其调用形式函数用来移动文件内部位置指针,其调用形式为:为: fseek(文件指针文件指针,
19、位移量位移量,起始点起始点);其中:其中:“文件指针文件指针”指向被移动的文件。指向被移动的文件。 “位移量位移量”表示移动的字节数,要求位移量是表示移动的字节数,要求位移量是long型数据,以便在文件长度大于型数据,以便在文件长度大于64KB 时不会出时不会出错。当用常量表示位移量时,要求加后缀错。当用常量表示位移量时,要求加后缀“L”。 “起始点起始点”表示从何处开始计算位移量,规定的表示从何处开始计算位移量,规定的起始点有三种:文件首,当前位置和文件尾。起始点有三种:文件首,当前位置和文件尾。 C语言程序设计与数据结构起始点起始点表示符号表示符号数字表示数字表示文件首文件首SEEK_SE
20、T0当前位置当前位置SEEK_CUR1文件末尾文件末尾SEEK_END2C语言程序设计与数据结构例如:例如:fseek(fp,100L,0); /* 将文件位置指针移到离文将文件位置指针移到离文件头件头100个字节处个字节处 */fseek(fp,20L,1); /* 将文件位置指针从当前位置将文件位置指针从当前位置后移后移20个字节的位置个字节的位置 */fseek(fp,-10L,SEEK_END); /* 将文件位置指针从文将文件位置指针从文件末尾前移件末尾前移10个字节的位置个字节的位置 */ 注意:注意:fseek函数一般用于二进制文件。在文本文函数一般用于二进制文件。在文本文件中由
21、于要进行转换,往往计算的位置会出现错误。件中由于要进行转换,往往计算的位置会出现错误。 C语言程序设计与数据结构 ftell函数函数: : 用来得到流式文件中的当前位置,用用来得到流式文件中的当前位置,用相对于文件开头的位移量来表示。如返回相对于文件开头的位移量来表示。如返回-1L则表则表示出错。示出错。 例如:例如:long a; a=ftell(fp); 文件结束检测函数文件结束检测函数feof函数函数feof( )判断文件是否处于文件结束位置,如文判断文件是否处于文件结束位置,如文件结束,则返回值为件结束,则返回值为1,否则为,否则为0。其调用格式为:其调用格式为: feof(文件指针文
22、件指针);说明:说明:文件结束标志文件结束标志EOF (每个文件末有一结束标每个文件末有一结束标志志EOF,值为值为-1)只适用于判断文本文件是否结束。只适用于判断文本文件是否结束。而函数而函数feof( )则对文本文件和二进制文件都适用。则对文本文件和二进制文件都适用。 C语言程序设计与数据结构10.4读写文件 10.4.1字符读写函数字符读写函数fgetc和和fputc 是以字符是以字符(字节字节)为单位的读写函数。每次可从文件读出或为单位的读写函数。每次可从文件读出或向文件写入一个字符。向文件写入一个字符。字符输入字符输入/ /出函数所处理的文件可以出函数所处理的文件可以是文本文件,也可
23、以是二进制文件。是文本文件,也可以是二进制文件。 10.4.2字符串读写函数字符串读写函数fgets和和fputs 处理的文件一般是文本文件,读写的数据以字符串为单位。处理的文件一般是文本文件,读写的数据以字符串为单位。10.4.3数据块读写函数数据块读写函数fread和和fwtrite 用于整块数据的读写函数。可用来读写一组数据,如一用于整块数据的读写函数。可用来读写一组数据,如一个数组、一个结构变量的值等。个数组、一个结构变量的值等。10.4.4格式化读写函数格式化读写函数fscanf和和fprintf 与前面使用的与前面使用的scanf和和printf 函数的功能相似,都是格式函数的功能
24、相似,都是格式化读写函数。两者的区别在于化读写函数。两者的区别在于fscanf函数和函数和fprintf函数的读函数的读写对象不是键盘和显示器,而是磁盘文件。写对象不是键盘和显示器,而是磁盘文件。 C语言程序设计与数据结构读字符函数fgetcfgetc函数的功能是从指定的文件中读取一个字符,函数调用函数的功能是从指定的文件中读取一个字符,函数调用的一般形式为:的一般形式为: 字符变量字符变量=fgetc(文件指针文件指针);例如:例如:ch=fgetc(fp); 其意义是从打开的文件其意义是从打开的文件fp中读取一个字符中读取一个字符并送入字符变量并送入字符变量ch中。中。对于对于fgetc函
25、数的使用有以下几点说明:函数的使用有以下几点说明:( (1) 在在fgetc函数调用中,读取的文件必须是以读或读写方式打函数调用中,读取的文件必须是以读或读写方式打开的。开的。( (2) 读取字符的结果也可以不向字符变量赋值,例如读取字符的结果也可以不向字符变量赋值,例如fgetc(fp); 但是读出的字符不能保存。但是读出的字符不能保存。 ( (3) 文件内部的位置指针在文件打开时总是指向文件的第一文件内部的位置指针在文件打开时总是指向文件的第一个字节。使用个字节。使用fgetc函数后,该位置指针将向后移动一个字函数后,该位置指针将向后移动一个字节。因此可连续多次使用节。因此可连续多次使用f
26、getc函数来读取多个字符。函数来读取多个字符。 C语言程序设计与数据结构【例【例10.1】将】将C盘根目录下的文本文件盘根目录下的文本文件test1.txt的内容在屏幕上输出。的内容在屏幕上输出。#includemain() FILE *fp; /* 定义了文件指针定义了文件指针fp */ char ch; if(fp=fopen(c:test1.txt,r)=NULL) /*以读文本文件方式打开并使以读文本文件方式打开并使fp指向该文件指向该文件*/ printf(nCannot open file, strike any key to exit!); getchar(); exit(1)
27、; ch=fgetc(fp); /* 先读出一个字符,然后进入循环先读出一个字符,然后进入循环 */ while(ch!=EOF) /* 判断文件是否结束判断文件是否结束 */ putchar(ch); /*把该字符显示在屏幕上把该字符显示在屏幕上*/ ch=fgetc(fp); /*再读入下一字符再读入下一字符 */ fclose(fp); printf(“n”); 本例程序的功能是从文件中逐个读取字符,并在屏幕上显示。如打开文件出错,给出提示并退出程序。只本例程序的功能是从文件中逐个读取字符,并在屏幕上显示。如打开文件出错,给出提示并退出程序。只要读出的字符不是文件结束标志要读出的字符不是
28、文件结束标志EOF,就把该字符显示在屏幕上,再读入下一字符。每读一次,文件就把该字符显示在屏幕上,再读入下一字符。每读一次,文件内部的位置指针向后移动一个字符,文件结束时,该指针指向内部的位置指针向后移动一个字符,文件结束时,该指针指向EOF。执行本程序将显示文件执行本程序将显示文件c:test1.txt的所有内容。的所有内容。 C语言程序设计与数据结构写字符函数fputc( ) fputc函数的功能是把一个字符写入指定的文件中,函数的功能是把一个字符写入指定的文件中,如写入成功则返回写入的字符,否则返回一个如写入成功则返回写入的字符,否则返回一个EOF,可用此来判断写入是否成功。可用此来判断
29、写入是否成功。 函数调用形式为:函数调用形式为: fputc(字符量,文件指针字符量,文件指针); /* 字符量是待写入的字符常量或变量字符量是待写入的字符常量或变量 */例如:例如:fputc( x, fp );其意义是把字符其意义是把字符x写入写入fp所指向的文件的位置指针的当前位置中。每写入一所指向的文件的位置指针的当前位置中。每写入一个字符,文件内部位置指针向后移动一个字节。个字符,文件内部位置指针向后移动一个字节。 被写入的文件可以用写、读写、追加方式打开。被写入的文件可以用写、读写、追加方式打开。 C语言程序设计与数据结构【例例10.2】 将从键盘上输入的一些字符(以将从键盘上输入
30、的一些字符(以“*”作为结束)写入作为结束)写入c盘根目录下盘根目录下名为名为test2.txt的文本文件中。的文本文件中。#include stdio.hmain( )FILE *p; char ch; if(p=fopen(c:test2.txt,w)=NULL) printf(File can not open!n); exit(0); printf(Please input,end input with * character:); while(ch=getchar()!=*) fputc(ch,p); fclose(p);printf(Write over!n );程序运行结束后,可
31、以在程序运行结束后,可以在c盘根目录下打开盘根目录下打开test2.txt查看其内容。查看其内容。思考一下,如果在写入完毕后想马上把刚才输入的内容打印在屏幕上该如何实现?思考一下,如果在写入完毕后想马上把刚才输入的内容打印在屏幕上该如何实现? C语言程序设计与数据结构字符串输入/出函数 字符串输入字符串输入/出函数所处理的文件一般是文本文件,读写的出函数所处理的文件一般是文本文件,读写的数据以字符串为单位。数据以字符串为单位。1. 读字符串函数读字符串函数fgets功能是从指定的文件中读一个字符串到字符数组中功能是从指定的文件中读一个字符串到字符数组中. .函数调用的形式为:函数调用的形式为:
32、 fgets(字符数组名字符数组名, n, 文件指针文件指针);其中其中n是一个正整数。该函数表示从文件中读出是一个正整数。该函数表示从文件中读出n-1个字符个字符(如在读满(如在读满n-1个字符之前就遇到了换行符或个字符之前就遇到了换行符或EOF,则读出则读出提前结束),并在读取的最后一个字符后加上串结束标志提前结束),并在读取的最后一个字符后加上串结束标志0。fgets函数的返回值是字符数组的首地址。读取字符串函数的返回值是字符数组的首地址。读取字符串后文件位置指针也自动后移若干个位置。后文件位置指针也自动后移若干个位置。例如:例如:fgets(str,n,fp); 意义是从意义是从fp所
33、指的文件中读出所指的文件中读出n-1个字符送入字符数组个字符送入字符数组str中。中。 C语言程序设计与数据结构【例【例10.3】从】从test2.txt文件中读入文件中读入8个字符组成一个字符串。个字符组成一个字符串。#includemain() FILE *fp; char arr20; if(fp=fopen(c:test2.txt ,r)=NULL) printf(nCannot open file,strike any key exit!); getchar(); exit(1); fgets(arr,9,fp); printf(n%sn,arr); fclose(fp);本例定义了
34、一个字符数组本例定义了一个字符数组str共共20个字节,在以读文本文件方式打开文件个字节,在以读文本文件方式打开文件string后,从中读出后,从中读出8个字符送入个字符送入str数组,在第数组,在第9个数组元素个数组元素str8内加上内加上0,然后,然后在屏幕上显示输出在屏幕上显示输出str数组存放的字符串。输出的八个字符正是例数组存放的字符串。输出的八个字符正是例10.2的前的前八个字符。八个字符。 C语言程序设计与数据结构2.写字符串函数写字符串函数fputsfputs函数的功能是向指定的文件写入一个字符串函数的功能是向指定的文件写入一个字符串(不写入字符串结束标记(不写入字符串结束标记
35、0),当成功写入一个),当成功写入一个字符串后,文件的位置指针会自动后移,函数返回字符串后,文件的位置指针会自动后移,函数返回值为值为0;否则,返回;否则,返回EOF(符号常量,其值为符号常量,其值为-1)。)。其调用形式为:其调用形式为: fputs( 字符串字符串, 文件指针文件指针 ); 其中字符串可以是字符串常量,也可以是字符数组名其中字符串可以是字符串常量,也可以是字符数组名或指针变量。或指针变量。 例如:例如:fputs(“a1b2”, fp) 是把字符串是把字符串“a1b2”写入写入fp所指的文件之中。所指的文件之中。 C语言程序设计与数据结构【例【例10.4】在文件】在文件te
36、st2.txt中追加一个不超过中追加一个不超过20个字符的字符串。个字符的字符串。#includemain() FILE *fp; char ch,st21; int count; if(fp=fopen(c:test2.txt ,a+)=NULL) printf(Cannot open file strike any key exit!); getchar(); exit(1); printf(input a string:n); gets (st); /* 输入不超过输入不超过20个字符的字符串暂放入数组个字符的字符串暂放入数组st中中 */ fputs(st,fp); rewind(fp
37、); /* 文件内部位置指针移到文件的开始位置文件内部位置指针移到文件的开始位置 */ printf(“n All of this file is: n”); ch=fgetc(fp); while(ch!=EOF) putchar(ch); ch=fgetc(fp); printf(n); fclose(fp); 本例要求在本例要求在test2.txt文件末添加一个字符串,因此,在程序第文件末添加一个字符串,因此,在程序第7行以追加读写文本文件的方式打开文件行以追加读写文本文件的方式打开文件test2.txt。然后输入字然后输入字符串暂放入数组符串暂放入数组st中,并用中,并用fputs函数
38、把该串写入文件函数把该串写入文件test2.txt。后在程序中用后在程序中用rewind函数把文件内部位置指针移到文件首,函数把文件内部位置指针移到文件首,再进入循环逐个显示当前文件中的全部内容。要注意用再进入循环逐个显示当前文件中的全部内容。要注意用scanf(“%s”,st)和和gets (st)输入字符串时的不同。输入字符串时的不同。 思考:如果不用思考:如果不用EOF,而是用而是用feof()来判断文件结束,上述阴影部分程序该如何改写?来判断文件结束,上述阴影部分程序该如何改写? C语言程序设计与数据结构数据块读写函数fread和fwtrite 语言还提供了用于整块数据的读写函数。可用
39、来读写语言还提供了用于整块数据的读写函数。可用来读写一组数据,如一个数组、一个结构变量的值等。一组数据,如一个数组、一个结构变量的值等。读数据块函数调用的一般形式为:读数据块函数调用的一般形式为:fread (buffer, size, count, fp);写数据块函数调用的一般形式为:写数据块函数调用的一般形式为:fwrite (buffer, size, count,fp); 其中:其中: size 表示每个数据块的字节数。表示每个数据块的字节数。 count 表示要读写的数据块块数。表示要读写的数据块块数。 fp 表示文件指针。表示文件指针。 buffer 是一个指针。是一个指针。 C
40、语言程序设计与数据结构 fread (buffer,size,count,fp) 的功能是从的功能是从fp所指向文件的位置所指向文件的位置指针的当前位置读取指针的当前位置读取count块数据(每个数据所占的字节数块数据(每个数据所占的字节数为为size),),共组成共组成count块长度为块长度为size的数据存入的数据存入butter所指所指定的内存空间中,当正确的读取了定的内存空间中,当正确的读取了count块数据后,文件内块数据后,文件内部指针会自动后移部指针会自动后移count*size个字节的位置,若数据存入正个字节的位置,若数据存入正确,则返回确,则返回count值,否则返回值,否
41、则返回NULL(符号常量,其值为符号常量,其值为0)。)。 例如:例如:fread(fa,4,5,fp); 其意义是从其意义是从fp所指的文件中,每次读所指的文件中,每次读4个字节个字节(一个实数一个实数)送入实型数组送入实型数组fa中,连续读中,连续读5次,即读次,即读5个实数到个实数到fa中。中。 fwrite (buffer,size,count,fp ) 表示将从表示将从buffer起始地址的起始地址的count块数据(每块数据的字节数为块数据(每块数据的字节数为size)写入到写入到fp所指向的所指向的文件中,当正确的写入文件中,当正确的写入count块数据后,文件位置指针会自块数据
42、后,文件位置指针会自动后移动后移count*size个字节的位置。若数据写入正确,则返回个字节的位置。若数据写入正确,则返回count值;否则返回值;否则返回NULL(符号常量,其值为符号常量,其值为0)。)。C语言程序设计与数据结构【例【例10.5】 从键盘上输入从键盘上输入10个整数,并把这些整数写入个整数,并把这些整数写入c:盘根目录下名为盘根目录下名为test3.dat的二进制文件中。的二进制文件中。#include stdio.hmain( )FILE *fp; int data10,i; if(fp=fopen(c:test3.dat,wb)=NULL) printf(File c
43、an not open!n); exit(0); printf(“Input ten integers: ”); for(i=0;i10;i+) scanf(%d, &datai); fwrite(data,sizeof(int),10,fp); fclose(fp);printf(“Write over! n”); C语言程序设计与数据结构【例【例10.6】 从从c:盘根目录下名为盘根目录下名为test3.dat的二进制文件中读取的二进制文件中读取10个整数,要求先读取第个整数,要求先读取第610个整数,再读取第个整数,再读取第15个整数。并把它们输出到屏幕上显示。个整数。并把它们输
44、出到屏幕上显示。#include stdio.hmain( )FILE *fp; int data10,i; if(fp=fopen(c:test3.dat,rb)=NULL) printf(File can not open!n); exit(0); fseek(fp, 5L*sizeof(int) ,0); /*先把文件位置指针后移先把文件位置指针后移*/ fread(data, sizeof(int),5, fp); /*先读取第先读取第610个整数放入个整数放入data0data4*/rewind(fp); /*把文件位置指针移到文件开始把文件位置指针移到文件开始*/fread(dat
45、a+5, sizeof(int),5, fp); /*再读取第再读取第15个整数放入个整数放入data5data9*/fclose(fp); for(i=0;i10;i+) printf(%5d,datai); C语言程序设计与数据结构【例【例10.7】从键盘输入三个学生数据,写入一个文件中,再读出这两个学生的数据显示在屏】从键盘输入三个学生数据,写入一个文件中,再读出这两个学生的数据显示在屏幕上。幕上。#includestruct stu char name20; int num; int age; char addr30;boya3,boyb3,*pp,*qq;main() FILE *f
46、p; int i; pp=boya; qq=boyb; if(fp=fopen(c:studata,wb+)=NULL) printf(Cannot open file strike any key exit!); getchar(); exit(1); C语言程序设计与数据结构for(i=0;iname,&pp-num,&pp-age,pp-addr); pp=boya; fwrite(pp,sizeof(struct stu),3,fp); rewind(fp); fread(qq,sizeof(struct stu),3,fp); fclose(fp); printf(n
47、nnametnumber age addrn); for(i=0;iname,qq-num,qq-age,qq-addr); 本例程序定义了一个结构本例程序定义了一个结构stu,说明了两个结构数组说明了两个结构数组boya和和boyb以及两个结构指针以及两个结构指针变量变量pp和和qq。pp指向指向boya,qq指向指向boyb。程序首先以读写方式打开二进制文程序首先以读写方式打开二进制文件件“studata”,输入三个学生数据之后,写入该文件中。然后把文件内部位置输入三个学生数据之后,写入该文件中。然后把文件内部位置指针移到文件首,读出三块学生数据后,在屏幕上显示。指针移到文件首,读出三块学
48、生数据后,在屏幕上显示。 C语言程序设计与数据结构格式化读写函数fscanf和fprintf fscanf函数,函数,fprintf函数与前面使用的函数与前面使用的scanf和和printf 函数的功能相似,都是格式化读写函数。两函数的功能相似,都是格式化读写函数。两者的区别在于者的区别在于fscanf函数和函数和fprintf函数的读写对象函数的读写对象不是键盘和显示器,而是磁盘文件。这两个函数的不是键盘和显示器,而是磁盘文件。这两个函数的调用格式为:调用格式为: fscanf(文件指针文件指针,格式字符串格式字符串,输入表列输入表列); fprintf(文件指针文件指针,格式字符串格式字符
49、串,输出表列输出表列); 例如:例如: fscanf(fp,%d%s,&i,s); fprintf(fp,%d%c,j,ch); 用用fscanf和和fprintf函数也可以完成例函数也可以完成例10.7的问题。的问题。修改后的程序如例修改后的程序如例10.8所示。所示。 C语言程序设计与数据结构【例【例10.8】用】用fscanf和和fprintf函数解决例函数解决例10.7的问题。的问题。#includestruct stu char name20; int num; int age; char addr30;boya3,boyb3,*pp,*qq;main( ) FILE *fp
50、;int i;pp=boya;qq=boyb;if(fp=fopen(c:studata,wb+)=NULL)printf(Cant open file strike any key exit!);getch();exit(1);C语言程序设计与数据结构printf(ninput datan);for(i=0;iname,&pp-num,&pp-age,pp-addr);pp=boya;for(i=0;iname,pp-num,pp-age,pp-addr);rewind(fp);for(i=0;iname,&qq-num,&qq-age,qq-addr);pr
51、intf(nnnametnumber age addrn);qq=boyb;for(i=0;iname,qq-num, qq-age,qq-addr);fclose(fp); 本程序中本程序中fscanf和和fprintf函数每次只能读写一个结构体数组元素,因此采用函数每次只能读写一个结构体数组元素,因此采用了循环语句来读写全部数组元素。了循环语句来读写全部数组元素。 还要注意指针变量还要注意指针变量pp,qq。由于循环改变了由于循环改变了它们的值,因此在程序中分别对它们重新赋予了数组的首地址。它们的值,因此在程序中分别对它们重新赋予了数组的首地址。 C语言程序设计与数据结构【例【例10.9】
52、 把命令行参数中的前一个文件名标识的文件,复制到后一个文件名标识的文件中,把命令行参数中的前一个文件名标识的文件,复制到后一个文件名标识的文件中, 如命令行中只有一个文件名则把该文件写到标准输出文件如命令行中只有一个文件名则把该文件写到标准输出文件(显示器显示器)中。中。#includemain(int argc,char *argv) FILE *fp1,*fp2; char ch; if(argc=1) /*如命令行参数中没有给出文件名,则给出提示信息如命令行参数中没有给出文件名,则给出提示信息*/ printf(Have not enter file name, strike any k
53、ey exit); getchar(); exit(0); if(fp1=fopen(argv1,rt)=NULL) printf(Cannot open %sn,argv1); getchar(); exit(1); .C语言程序设计与数据结构if(argc=2) /*如果只给出一个文件名,则使如果只给出一个文件名,则使fp2指向标准输出文件指向标准输出文件(即显示器即显示器)*/fp2=stdout; else if(fp2=fopen(argv2,wt+)=NULL) printf(Cannot open %sn,argv1); getchar(); exit(1); while(ch=
54、fgetc(fp1)!=EOF) /*用循环语句逐个读出文件用循环语句逐个读出文件1中的字符再送到文件中的字符再送到文件2中中*/ fputc(ch,fp2); fclose(fp1); fclose(fp2); 本程序为带参数的本程序为带参数的main函数。程序中定义了两个文件指针函数。程序中定义了两个文件指针fp1和和fp2,分别指向命令行参分别指向命令行参数中给出的文件。如命令行参数中没有给出文件名,则给出提示信息。如果只给出一个数中给出的文件。如命令行参数中没有给出文件名,则给出提示信息。如果只给出一个文件名,则使文件名,则使fp2指向标准输出文件指向标准输出文件(即显示器即显示器)。
55、然后用循环语句逐个读出文件。然后用循环语句逐个读出文件1中的字符中的字符再送到文件再送到文件2中。假设项目名称为中。假设项目名称为cexam1_1,则生成一个可执行文件则生成一个可执行文件cexam1_1.exe。在在DOS下运行结果如图所示:下运行结果如图所示: C语言程序设计与数据结构第一次运行第一次运行, 不指定文件名,系统提示不指定文件名,系统提示have not enter file name, strike any key exit。再次运行时,给出了一个文件名再次运行时,给出了一个文件名test1.txt,故输出给标准输出文件故输出给标准输出文件stdout,即在显示器上显示文件
56、即在显示器上显示文件test1.txt的内容的内容“1234”。第三次运行,给出了二个文件名,因此把第三次运行,给出了二个文件名,因此把test1.txt中的内容读出,写入到中的内容读出,写入到test9.txt之中。通过用之中。通过用DOS命令命令type显示,显示,可以看到可以看到test9.txt的内容也是的内容也是“1234”。 C语言程序设计与数据结构【例【例10.10】以下叙述中不正确的是以下叙述中不正确的是( )。)。( (A) CA) C语言中的文本文件以语言中的文本文件以 ASCII ASCII 码形式存储数据码形式存储数据( (B) CB) C语言中对二进制文件比对文本文件
57、的访问速度快语言中对二进制文件比对文本文件的访问速度快( (C) CC) C语言中语言中,随机读写方式不适用于文本文件随机读写方式不适用于文本文件( (D) CD) C语言中语言中,顺序读写方式不适用于二进制文件顺序读写方式不适用于二进制文件分析分析: 文本文件的每个字节存放一个文本文件的每个字节存放一个 ASCII ASCII 码码,代表一个字符代表一个字符。所以所以,答案答案A A是正确的。数据在计算机中是以二进制形式存放的是正确的。数据在计算机中是以二进制形式存放的,当计算机与当计算机与二进制文件进行数据交换时二进制文件进行数据交换时,不需要进行转换不需要进行转换,所以访问速度比文本文所
58、以访问速度比文本文件要快件要快。在文本文件中在文本文件中,数据是以数据是以ASCIIASCII码形式存放的码形式存放的,不同数据占用不同数据占用的空间是不同的的空间是不同的,不利于随机读写方式的文件位置指针移动不利于随机读写方式的文件位置指针移动,所以随机所以随机读写方式不适用于读写方式不适用于文文本文件本文件,答案答案C C是正确的是正确的。二进制文件中二进制文件中,一个数一个数据占用的字节数是固定的据占用的字节数是固定的,所以所以既既可以使用随机读写方式也可可以使用随机读写方式也可以以使用顺使用顺序读写方式序读写方式,所以答案所以答案D D的说法不正确。的说法不正确。答案答案: : D D
59、 C语言程序设计与数据结构【例【例10.11】 若要打开若要打开A A盘上盘上useruser子目录下名为子目录下名为abc.txtabc.txt的文本文件进行读、写操作的文本文件进行读、写操作,下面符合下面符合此要求的函数调用是此要求的函数调用是( )。)。(A)fopen(“A:userabc.txt”, “r”) (B) (B) fopen(“A:userabc.txt”,“r+”) (C) fopen(“A:userabc.txt”, “rb”) (D) fopen(“A:userabc.txt”,“w”) )分析分析: : 本题考查的是本题考查的是fopen()fopen()函数的应
60、用函数的应用。在写第一在写第一实参的路径名时实参的路径名时,应使用应使用“”“”表示表示“”“”。若需。若需要对打开的文本文件进行读写要对打开的文本文件进行读写,应使用读写模式应使用读写模式“r+”r+”。答案答案 : :B B C语言程序设计与数据结构【例【例10.1210.12】下面程序把从终端读入的文本下面程序把从终端读入的文本( ( 用用 作为文本结束标志作为文本结束标志 ) ) 复制到一个名为复制到一个名为bi.dat的新文件中的新文件中,请填空。请填空。#include FILE *fp;main( )char ch;if(fp=fopen(_(1)_)=NULL) exit(0)
61、;while(ch=getchar()!=)fputc(ch,fp);_(2)_分析:分析: 本题考查的是文件的打开和关闭。由于题目要求将文本复制到本题考查的是文件的打开和关闭。由于题目要求将文本复制到bi.datbi.dat文件文件中中,所以需要以写方式打开所以需要以写方式打开bi.batbi.bat文件文件, ,所以所以, ,第一个空中填写第一个空中填写fopen()函数的参函数的参数应是数应是bi.bat,W。文件打开成功后文件打开成功后,whilewhile循环中不断地读入字符循环中不断地读入字符,并并写入写入bi.batbi.bat文件中文件中,数据输入结束后数据输入结束后,应及时关闭文件。所以第二应及时关闭文件。所以第二个个空中填写空中填写fclose(fp);答案答案: : (1)“bi.bat”,“W” (2) fclose(fp); C语言程序设计与数据结构谢谢!谢谢!