求解C语言fopen函数的读写模式

求解C语言fopen函数的读写模式,第1张

r模式打开的文件一般都是ascii文件,也就是文件文件,供直接阅读的(用记事本程序打开该文件可以直接阅读),这种方式下打开文件一般用fscanf()/fgetc()等函数 *** 作

rb模式打开的文件一般不用于直接阅读(用记事本打开后看到的可能是乱码),而是存取程序要用到的二进制数据,这种方式下打开文件一般用fread()函数进行 *** 作

fopen函数用来打开一个文件,其调用的一般形式为:

文件指针名=fopen(文件名,使用文件方式)

其中,

“文件指针名”必须是被说明为FILE 类型的指针变量;

“文件名”是被打开文件的文件名;

“使用文件方式”是指文件的类型和 *** 作要求。

“文件名”是字符串常量或字符串数组。

例如:

FILE *fp;

fp=("file a","r")

其意义是在当前目录下打开文件file a,只允许进行“读” *** 作,并使fp指向该文件。

又如:

FILE *fphzk

fphzk=("c:\\hzk16","rb")

其意义是打开C驱动器磁盘的根目录下的文件hzk16,这是一个二进制文件,只允许按二进制方式进行读 *** 作。两个反斜线“\\ ”中的第一个表示转义字符,第二个表示根目录。

使用文件的方式共有12种,下面给出了它们的符号和意义。

文件使用方式

意义

“rt”

只读打开一个文本文件,只允许读数据

“wt”

只写打开或建立一个文本文件,只允许写数据

“at”

追加打开一个文本文件,并在文件末尾写数据

“rb”

只读打开一个二进制文件,只允许读数据

“wb”

只写打开或建立一个二进制文件,只允许写数据

“ab”

追加打开一个二进制文件,并在文件末尾写数据

“rt+”

读写打开一个文本文件,允许读和写

“wt+”

读写打开或建立一个文本文件,允许读写

“at+”

读写打开一个文本文件,允许读,或在文件末追加数据

“rb+”

读写打开一个二进制文件,允许读和写

“wb+”

读写打开或建立一个二进制文件,允许读和写

“ab+”

读写打开一个二进制文件,允许读,或在文件末追加数据

对于文件使用方式有以下几点说明:

1) 文件使用方式由r,w,a,t,b,+六个字符拼成,各字符的含义是:

r(read): 读

w(write): 写

a(append): 追加

t(text): 文本文件,可省略不写

b(banary): 二进制文件

+: 读和写

2) 凡用“r”打开一个文件时,该文件必须已经存在,且只能从该文件读出。

3) 用“w”打开的文件只能向该文件写入。若打开的文件不存在,则以指定的文件名建立该文件,若打开的文件已经存在,则将该文件删去,重建一个新文件。

4) 若要向一个已存在的文件追加新的信息,只能用“a”方式打开文件。但此时该文件必须是存在的,否则将会出错。

5) 在打开一个文件时,如果出错,fopen将返回一个空指针值NULL。在程序中可以用这一信息来判别是否完成打开文件的工作,并作相应的处理。因此常用以下程序段打开文件:

6) if((fp=fopen("c:\\hzk16","rb")==NULL)

{

printf("\nerror on open c:\\hzk16 file!")

getch()

exit(1)

}

这段程序的意义是,如果返回的指针为空,表示不能打开C盘根目录下的hzk16文件,则给出提示信息“error on open c:\ hzk16 file!”,下一行getch()的功能是从键盘输入一个字符,但不在屏幕上显示。在这里,该行的作用是等待,只有当用户从键盘敲任一键时,程序才继续执行,因此用户可利用这个等待时间阅读出错提示。敲键后执行exit(1)退出程序。

7) 把一个文本文件读入内存时,要将ASCII码转换成二进制码,而把文件以文本方式写入磁盘时,也要把二进制码转换成ASCII码,因此文本文件的读写要花费较多的转换时间。对二进制文件的读写不存在这种转换。

8) 标准输入文件(键盘),标准输出文件(显示器),标准出错输出(出错信息)是由系统打开的,可直接使用。

本文分别简单讲述了如何利用C和C++库函数对文件的读写,涵盖了常用的文件 *** 作函数。网上关于C/C++文件 *** 作的介绍的博客很多,所以本文也大同小异,也主要是做一个备忘,不有每次都百度了。但本文重点在于几个库函数的运用,而不是从文件的属性或分类开始。

一、C文件 *** 作:

C语言中,文件的 *** 作都是通过一个FILE类型的文件指针进行,也就是说只有通过文件指针,才能调用相应的文件。FILE是一个由系统定义的结构体(定义在stdio.h中),可以存放文件的相关信息。

文件的打开(fopen函数)

文件的 *** 作过程一般为:打开 =>读/写 =>关闭。

fopen函数的原型如下:

FILE* fopen(const char* filename, const char* mode)

函数调用就比较简单了:

FILE* fp

fp = fopen("file1.txt", "r")

如果打开成功,返回file1.txt文件的指针,如果打开失败,返回一个NULL指针,所以调用fopen()后我们需要检查fp的值才进行下一步 *** 作。fopen()的第一个参数为文件名,第二个参数为文件打开方式,含义如下表:

说明: 1. 使用"r"时,如果文件不存在,则出错。

2. 使用"w"时,如果没有文件,则创建一个新文件。

3. 使用"a"时,如果希望向文件尾添加数据,则该文件必须存在,否则出错。

4. "r+","w+","a+"都是可以输入和输出数据,但必须遵守上述3点

5. *** 作二进制文件时,加上"b"字符,且二进制文件对换行符不会进行转换,而文本文件会将换行符转换为回车和换行两个字符。

文件的关闭(fclose函数)

在使用完一个文件后,若不关闭则会造成系统资源泄漏。使用fclose()关闭文件即可,原型为 int fclose(FILE* fp)。使用:fclose(fp)flose()返回0时为顺利关闭文件,否则返回EOF(-1)。

文件的读写

1) fputc(), fgetc()分别为从文件流中写和读一个字符,原型分别如下:

写:int fputc(int c, FILE* fp) 读:int fgetc(FILE* fp)失败时均返回EOF

2) fputs(), fgets()分别为从文件流中写和读一个字符串,原型分别如下:

写:int fputs(const char* str, FILE* fp) 例如: fputs("I love this game!", fp)

读:char* fgets(char* str, int n, FILE* fp)从流中读取n-1个字符或读完一行,参数str用于接收读取的字符串。注意当读取一行时,不包括行尾的'\n'字符。

3) fseek() 一般用于二进制模式打开的文件中,功能是定位到流中指定的位置。原型如下:

int fseek(FILE* fp, lont offset, int whence)参数offset是移动的字符数,whence是移动的基准,取值是:

SEEK_SET 0 //文件开头

SEEK_CUR 1 //当前读写的位置

SEEK_END 2 //文件尾部

4) fprintf(),fscanf()是将数据按格式输出输入到文件流中,用法类似printf()和scanf()。原型分别如下:

int fprintf(FILE* fp, const char* format, ...)它与printf()不同的就是将数据写到了文件流中,而不是控制台罢了。

int fscanf(FILE* fp, cosnt char* format, ...)从文件流中按格式读取,与scanf()不同的就是数据是从文件流中读取而已。

例如: fprintf(fp, "count=%d", 5)fscanf(fp, "%d", &x)

5) feof()是检测是否已到文件尾,是返回真,否则返回0,原型是 int feof(FILE* fp)

6) rewind() 则是把当前的读写位置回到文件开始,相当于 fseek(fp, 0L, SEEK_SET)原型: void rewind(FILE* fp)

7) remove() 删除文件,原型: int remove(const char* filename) 参数为要删除的文件名,成功则返回0

8) fread(), fwrite() 它们相当于可将一块的数据读出或写入,相当的方便。原型如下:

size_t fread(void* ptr, size_t size, size_t n, FILE* fp)从流中读指定个数的字符,size是每块的字节娄,n则是读取的块数。

size_t fwrite(const void* ptr, size_t size, size_t n, FILE* fp)类似的是向文件流中写入n块size字节数的数据。可以看到数据指针为void*型,即可以使用任何类型的指针来替换。例如:

现在一个结构体: struct student_t{char name[16]int idint age}创建三个学生的数据并赋值:struct student_t stu[3]

这时,当我们找开文件后(一般是进制模式),可以调用fwrite()将三个学生的数据都写入到文件中,两种方式:

for(int i = 0i <3++i)

fwrite(&stu[i], sizeof(struct student_t), 1, fp)

或者:fwrite(stu, sizeof(struct student_t), 3, fp)

些时,我们调用fread()函数便可很轻松的将刚才写入的3个学生的数据读取出来:

struct student_t stus[3]

for(int i = 0i <3++i)

fread(&stus[i], sizeof(struct student_t), 1, fp)

或者:fread(stus, sizeof(struct student_t), 3, fp)这样便可将三个学生的数据读入到stus变量中了(有木有很方便呐)

注意:如果你发现使用fread()读取之后,最后一个学生读取的数据不完全,可能是由于你没有使用二进制模式打开的原因。

9) 最后是tmpfile()和tmpnam(),前者为生成一个临时文件,后者为生成一个唯一的文件名,具体使用在此不介绍了。

二、使用C++中的fstream文件流 *** 作类进行文件的读写

使用fstream *** 作文件与使用C库函数类似,只不过fstream为面向对象方式,或多了上些C++的特性。首先,这里大概有三个流:

fstream为文件输入输出流,ifstream为输入文件流,ofstream为输出文件流,它们与ostream不同的就流的目的地为文件,而不是控制台。这里只介绍与上述的一些不同点:

1. 打开文件,如可以是 ifstream input_file("file2.txt")这样将会以默认方式打开file2.txt文件并进行读取。也可使用open()方式打开一个文件,并指定打开方式,例如:

ifstream input_file

input_file.open("file2.txt", ios::binary)

打开后,可以使用is_open()检测是否打开成功:

if(input_file.is_open()){},然后可以使用流 *** 作符向文件写数据了,例如:

input_file <<"this is a test line"

input_file <<"another info"

另外,这里的文件打开方式在ios空间下:

ios::app 添加到文件尾

ios::ate 把文件标志放在末尾而非起始。

ios::trunc 默认. 截断并覆写文件。

ios::nocreate 文件不存在也不创建。

ios::noreplace 文件存在则失败。

ofstream使用方式类似,读取一行数据可以使用getline(buf, count), 类型于fgets()。fstream类还提供一个很多其它方法,如fclose()为关闭文件,eof()用于检测状态是否已经到了文件末尾。

这里还有两个类似于上述的fread()和fwrite()函数,是read(), write(),功能和用法类似类似,例如:

output_file.write((const char*)stu, 3 * sizeof(struct student_t))

input_file.read((char*)stus, 3 * sizeof(struct student_t))

注意,这里也需要使用二进制模式打开,否则read的时候最后的上些数据读不完全。调用上面两个函数后可以使用bad()来检测文件流对象是否错误,例如,if(input_file.bad()){printf("error when read file\n")return},最后input_file.close()即可。

最后,C/C++文件的 *** 作并不复杂,多使用几次便可熟练掌握,需要注意的就是文件的打开方式,和当用同一个文件指针进行又读又写时,注意文件指针位置的移动。


欢迎分享,转载请注明来源:内存溢出

原文地址: http://outofmemory.cn/tougao/11957033.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2023-05-20
下一篇 2023-05-20

发表评论

登录后才能评论

评论列表(0条)

保存