1.2 文件的输入输出函数
键盘、显示器、打印机、磁盘驱动器等逻辑设备, 其输入输出都能够通过文件管理的方法来完毕。而在编程时使用最多的要算是磁盘文件, 因此本节主要以磁盘文件为主, 具体介绍Turbo C2.0提供的文件操作函数, 当然这些对文件的操作函数也适合于非磁盘文件的情况。
另外, Turbo C2.0提供了两类关于文件的函数。一类称做标准文件函数也称缓冲型文件函数, 这是ANSI标准定义的函数; 还有一类叫非标准文件函数, 也称非缓冲型文件函数。这类函数最早公用于UNIX操作系统, 但如今MS-DOS3.0 以上版本号的操作系统也能够使用。以下分别进行介绍。 1.2.1 标准文件函数
标准文件函数主要包含文件的打开、关闭、读和写等函数。不象BASIC 、 FORTRAN语方有顺序文件和随机文件之分, 在打开时就应按不同的方式确定。 Turbo C2.0并不区分这两种文件, 但提供了两组函数, 即顺序读写函数和随机读写函数。
一、文件的打开和关闭
不论什么一个文件在使用之前和使用之后, 必需要进行打开和关闭, 这是由于操作系统对于同一时候打开的文件数目是有限制的, DOS操作系统中, 能够在DEVICE .SYS中定义同意同一时候打开的文件数n(用files=n定义)。当中n 为可同一时候打开的文件数, 一般n<=20。因此在使用文件前应打开文件, 才可对当中的信息进行存取。用完之后需要关闭, 否则将会出现一些意想不到的错误。Turbo C2.0提供了打开和关闭文件的函数。
1. fopen()函数
fopen函数用于打开文件, 其调用格式为:
FILE *fopen(char *filename, *type);
在介绍这个函数之;前, 先了解一下以下的知识。
(1) 流(stream)和文件(file)
流和文件 在Turbo C2.0中是有差别的, Turbo C2.0 为编程者和被訪问的设备之间提供了一层抽象的东西, 称之为"流", 而将详细的实际设备叫做文件。流是一个逻辑设备, 具有相同的行为。因此, 用来进行磁盘文件写的函数也相同能够用来进行打印机的写入。在Turbo C2.0中有两种性质的流: 文字流( text stream)和二进制(binary stream)。对磁盘来说就是文本文件和二进制文件。本软件为了便于让读者易理解Turbo C2.0语言而没有对流和文件作特别区分。
(2) 文件指针FILE
实际上FILE是一个新的数据类型。它是Turbo C2.0的基本数据类型的集合, 称之为结构指针。有关结构的概念将在第四节中具体介绍, 这里仅仅要将FILE理解为一个包含了文件管理有关信息的数据结构, 即在打开文件时必须先定义一个文件指针。
(3) 以后介绍的函数调用格式将直接写出形式參数的数据类型和函数返回值的数据类型。比如: 上面打开文件的函数, 返回一个文件指针, 当中形式參数有两个, 均为字符型变量(字符串数组或字符串指针)。本软件不再对函数的调用格式作具体说明。
如今再来看打开文件函数的使用方法。
fopen()函数中第一个形式參数表示文件名称, 能够包括路径和文件名称两部分。如:
"B:TEST.DAT"
"C://TC//TEST.DAT"
假设将路径写成"C:/TC/TEST.DAT"是不对的, 这一点要特别注意。
第二个形式參数表示打开文件的类型。关于文件类型的规定參见下表。
表 文件操作类型
━━━━━━━━━━━━━━━━━━━━━━━━━━━━
字符 含义
────────────────────────────
"r" 打开文字文件仅仅读
"w" 创建文字文件仅仅写
"a" 增补, 假设文件不存在则创建一个
"r+" 打开一个文字文件读/写
"w+" 创建一个文字文件读/写
"a+" 打开或创建一个文件增补
"b" 二进制文件(能够和上面每一项合用)
"t" 文这文件(默认项)
━━━━━━━━━━━━━━━━━━━━━━━━━━━━
假设要打开一个CCDOS子文件夹中, 文件名称为CLIB的二进制文件, 可写成:
fopen("c://ccdos//clib", "rb");
假设成功的打开一个文件, fopen()函数返回文件指针, 否则返回空指针 (NULL)。由此可推断文件打开是否成功。
2. fclose()函数
fclose()函数用来关闭一个由fopen()函数打开的文件 , 其调用格式为:
int fclose(FILE *stream);
该函数返回一个整型数。当文件关闭成功时, 返回0, 否则返回一个非零值。能够依据函数的返回值推断文件是否关闭成功。
例10:
#iclude<stdio.h>
main()
{
FILE *fp; /*定义一个文件指针*/
int i;
fp=fopen("CLIB", "rb"); /*打开当前文件夹名为CLIB的文件仅仅读*/
if(fp==NULL) /*推断文件是否打开成功*/
puts("File open error");/*提示打开不成功*/
i=fclose(fp); /*关闭打开的文件*/
if(i==0) /*推断文件是否关闭成功*/
printf("O,K"); /*提示关闭成功*/
else
puts("File close error");/*提示关闭不成功*/
} 二、有关文件操作的函数
本节所讲的文件读写函数均是指顺序读写, 即读写了一条信息后, 指针自己主动加1。以下分别介绍写操作函数和读操作函数。 1. 文件的顺序写函数
fprintf()、fputs()和fputc()函数
函数fprintf()、fputs()和fputc()均为文件的顺序写操作函数, 其调用格式例如以下:
int fprintf(FILE *stream, char *format, <variable-list>);
int fputs(char *string, FILE *steam);
int fputc(int ch, FILE *steam);
上述三个函数的返回值均为整型量。fprintf() 函数的返回值为实际写入文件里的字罕个数(字节数)。假设写错误, 则返回一个负数, fputs()函数返回0时表明将string指针所指的字符串写入文件里的操作成功, 返回非0时, 表明写操作失败。fputc()函数返回一个向文件所写字符的值, 此时写操作成功, 否则返回EOF(文件结束结束其值为-1, 在stdio.h中定义)表示写操作错误。
fprintf( ) 函数中格式化的规定与printf( ) 函数同样, 所不同的仅仅是 fprintf()函数是向文件里写入。而printf()是向屏幕输出。
以下介绍一个样例, 执行后产后一个test.dat的文件。
例11:
#include<stdio.h>
main()
{
char *s="That's good news"); /*定义字符串指针并初始化*/
int i=617; /*定义整型变量并初始化*/
FILE *fp; /*定义文件指针*/
fp=fopne("test.dat", "w"); /*建立一个文字文件仅仅写*/
fputs("Your score of TOEFLis", fp);/*向所建文件写入一串字符*/
fputc(':', fp); /*向所建文件写冒号:*/
fprintf(fp, "%d/n", i); /*向所建文件写一整型数*/
fprintf(fp, "%s", s); /*向所建文件写一字符串*/
fclose(fp); /*关闭文件*/
}
用DOS的TYPE命令显示TEST.DAT的内容例如以下所看到的:
屏幕显示
Your score of TOEFL is: 617
That's good news 2. 文件的顺序读操作函数
fscanf()、fgets()和fgetc()函数
函数fscanf()、fgets()和fgetc()均为文件的顺序读操作函数, 其调用格式例如以下:
int fscanf(FILE *stream, char *format, <address-list>);
char fgets(char *string, int n, FILE *steam);
int fgetc(FILE *steam);
fscanf()函数的使用方法与scanf()函数类似, 仅仅是它是从文件里读到信息。 fscanf()函数的返回值为EOF(即-1), 表明读错误, 否则读数据成功。fgets()函数从文件里读取至多n-1个字符(n用来指定字符数), 并把它们放入string指向的字符串中, 在读入之后自己主动向字符串未尾加一个空字符, 读成功返回string指针, 失败返回一个空指针。fgetc()函数返回文件当前位置的一个字符, 读错误时返回EOF。
以下程序读取例11产生的test.dat文件, 并将读出的结果显示在屏幕上。
例12
#include<stdio.h>
main()
{
char *s, m[20];
int i;
FILE *fp;
fp=fopen("test.dat", "r"); /*打开文字文件仅仅读*/
fgets(s, 24, fp); /*从文件里读取23个字符*/
printf("%s", s); /*输出所读的字符串*/
fscanf(fp, "%d", &i); /*读取整型数*/
printf("%d", i); /*输出所读整型数*/
putchar(fgetc(fp)); /*读取一个字符同一时候输出*/
fgets(m, 17, fp); /*读取16个字符*/
puts(m); /*输出所读字符串*/
fclose(fp); /*关闭文件*/
getch(); /*等待任一键*/
}
执行后屏幕显示:
Your score of TOEFL is: 617
That's good news
假设将上例中fscanf(fp, "%d", &i)改为fscanf(fp, "%s", m), 再将其后的输出语句改为printf("%s", m), 则可得出相同的结果。由此可见Turbo C2. 0 中仅仅要是读文字文件, 则不论是字符还是数字都将按其ASCII值处理。 另外还要说明的一点就是fscanf()函数读到空白符时, 便自己主动结束, 在使用时要特别注意。 3. 文件的随机读写
有时用户想直接读取文件里间某处的信息, 若用文件的顺序读写必须从文件头開始直到要求的文件位置再读, 这显然不方便。Turbo C2.0提供了一组文件的随机读写函数, 即能够将文件位置指针定位在所要求读写的地方直接读写。
文件的随机读写函数例如以下:
int fseek (FILE *stream, long offset, int fromwhere);
int fread(void *buf, int size, int count, FILE *stream);
int fwrite(void *buf, int size, int count, FILE *stream);
long ftell(FILE *stream);
fseek()函数的作用是将文件的位置指针设置到从fromwhere開始的第offset 字节的位置上, 当中fromwhere是下列几个宏定义之中的一个:
文件位置指针起始计算位置fromwhere ━━━━━━━━━━━━━━━━━━━━━━━━━━━
符号常数 数值 含义 ───────────────────────────
SEEK_SET 0 从文件开头
SEEK_CUR 1 从文件指针的现行位置
SEEK_END 2 从文件末尾 ━━━━━━━━━━━━━━━━━━━━━━━━━━━
offset是指文件位置指针从指定開始位置(fromwhere指出的位置)跳过的字节数。它是一个长整型量, 以支持大于64K字节的文件。fseek()函数一般用于对二进制文件进行操作。
当fseek()函数返回0时表明操作成功, 返回非0表示失败。
以下程序从二进制文件test_b.dat中读取第8个字节。
例13:
#include<stdio.h>
main()
{
FILE *fp;
if((fp=fopen("test_b.dat", "rb"))==NULL)
{
printf("Can't open file");
exit(1);
}
fseek(fp, 8. 1, SEEK_SET);
fgetc(fp);
fclose(fp);
}
fread()函数是从文件里读count个字段, 每一个字段长度为size个字节, 并把它们存放到buf指针所指的缓冲器中。
fwrite()函数是把buf指针所指的缓冲器中, 长度为size个字节的count个字段写到stream指向的文件里去。
随着读和写字节数的增大, 文件位置指示器也增大, 读多少个字节, 文件位置指示器对应也跳过多少个字节。读写完成函数返回所读和所写的字段个数。
ftell()函数返回文件位置指示器的当前值, 这个值是指示器从文件头開始算起的字节数, 返回的数为长整型数, 当返回-1时, 表明出现错误。
以下程序把一个浮点数组以二进制方式写入文件test_b.dat中。
例14:
#include <stdio.h>
main()
{
float f[6]={3.2, -4.34, 25.04, 0.1, 50.56, 80.5};
/*定义浮点数组并初始化*/
int i;
FILE *fp;
fp=fopen("test_b.dat", "wb"); /*创建一个二进制文件仅仅写*/
fwrite(f, sizeof(float), 6, fp);/*将6个浮点数写入文件里*/
fclose(fp); /*关闭文件*/
}
以下样例从test_b.dat文件里读100个整型数, 并把它们放到dat数组中。
例15:
#include <stdio.h>
main()
{
FILE *fp;
int dat[100];
fp=fopen("test_b.dat", "rb");/*打开一个二进制文件仅仅读*/
if(fread(dat, sizeof(int), 100, fp)!=100)
/*推断是否读了100个数*/
{
if(feof(fp))
printf("End of file"); /*不到100个数文件结束*/
else
printf("Read error"); /*读数错误*/
fclose(fp); /*关闭文件*/
}
注意:
当用标准文件函数对文件进行读写操作时, 首先将所读写的内容放进缓冲区, 即写函数仅仅对输出缓冲区进行操作, 读函数仅仅对输入缓冲区进行操作。比如向一个文件写入内容, 所写的内容将首先放在输出缓冲区中, 直到输出缓冲区存满或使用fclose()函数关闭文件时, 缓冲区的内容才会写入文件里。若无fclose() 函数, 则不会向文件里存入所写的内容或写入的文件内容不全。有一个对缓冲区进行刷新的函数, 即fflush(), 其调用格式为:
int fflush(FILE *stream);
该函数将输出缓冲区的内容实际写入文件里, 而将输入缓冲区的内容清除掉。 4. feof()和rewind()函数
这两个函数的调用格式为:
int feof(FILE *stream);
int rewind(FILE *stream);
feof()函数检測文件位置指示器是否到达了文件结尾, 若是则返回一个非0 值, 否则返回0。这个函数对二进制文件操作特别实用, 由于二进制文件里, 文件结尾标志EOF也是一个合法的二进制数, 仅仅简单的检查读入字符的值来推断文件是否结束是不行的。假设那样的话, 可能会造成文件未结尾而被觉得结尾, 所以就必须有feof()函数。
以下的这条语句是经常使用的推断文件是否结束的方法。
while(!feof(fp))
fgetc(fp);
while为循环语句, 将在以下介绍。
rewind()函数用于把文件位置指示器移到文件的起点处, 成功时返回0, 否则, 返回非0值。 1.2.2 非标准文件函数
这类函数最早用于UNIX操作系统, ANSI标准没有定义, 但有时也经经常使用到, DOS 3.0以上版本号支持这些函数。它们的头文件为io.h。
一、文件的打开和关闭
1. open()函数
open()函数的作用是打开文件, 其调用格式为:
int open(char *filename, int access);
该函数表示按access的要求打开名为filename的文件, 返回值为文件描写叙述字, 当中access有两部分内容: 基本模式和修饰符, 两者用" "("或")方式连接。修饰符能够有多个, 但基本模式仅仅能有一个。access的规定如表3-2。
表3-2 access的规定 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━ 基本模式 含义 修饰符 含 义 ──────────────────────────── O_RDONLY 仅仅读 O_APPEND 文件指针指向末尾 O_WRONLY 仅仅写 O_CREAT 文件不存在时创建文件,
属性按基本模式属性 O_RDWR 读写 O_TRUNC 若文件存在, 将其长度
缩为0, 属性不变
O_BINARY 打开一个二进制文件
O_TEXT 打开一个文字文件 ━━━━━━━━━━━━━━━━━━━━━━━━━━━━
open()函数打开成功, 返回值就是文件描写叙述字的值(非负值), 否则返回-1。 2. close()函数
close()函数的作用是关闭由open()函数打开的文件, 其调用格式为:
int close(int handle);
该函数关闭文件描写叙述字handle相连的文件。 二、读写函数
1. read()函数
read()函数的调用格式为:
int read(int handle, void *buf, int count);
read()函数从handle(文件描写叙述字)相连的文件里, 读取count个字节放到buf 所指的缓冲区中, 返回值为实际所读字节数, 返回-1表示出错。返回0 表示文件结束。 2. write()函数
write()函数的调用格式为:
int write(int handle, void *buf, int count);
write()函数把count个字节从buf指向的缓冲区写入与handle相连的文件里, 返回值为实际写入的字节数。 三、随机定位函数
1. lseek()函数
lseek()函数的调用格式为:
int lseek(int handle, long offset, int fromwhere);
该函数对与handle相连的文件位置指针进行定位, 功能和使用方法与fseek() 函数同样。 2. tell()函数
tell()函数的调用格式为:
long tell(int handle);
该函数返回与handle相连的文件现生位置指针, 功能和使用方法与ftell()同样。