C语言入门系列之11.文件和文件操作

在命令行中封装表白送给她

今天520,是一个表白的好日子,虽然现在还是一条孤独的单身狗,但还是蹭一波热度吧😜你也来试一下,给心爱的她一个神秘礼物吧。

一、C文件概述

1.基本概念

文件是指一组相关数据的有序集合,这个数据集有一个名称,叫做文件名。
我们在前面的已经使用到了很多文件,例如源程序文件、目标文件、可执行文件、库文件 (头文件)等。

文件通常是驻留在外部介质(如磁盘等)上的,在使用时才调入内存中来。

操作系统是以文件为单位对数据进行管理的,示意如下:
程序处理文件示意

2.文件分类

从不同的角度可对文件作不同的分类:
(1)从用户的角度看,文件可分为:

  • 特殊文件(标准输入输出文件或标准设备文件)
  • 普通文件(磁盘文件)

(2)从操作系统的角度看,每一个与主机相连的输入、输出设备都看作是一个文件。
例如:

  • 输入文件,终端键盘等
  • 输出文件,显示屏和打印机等

(3)按数据的组织形式:

  • ASCII文件(文本文件)
    每一个字节放一个ASCII代码。
  • 二进制文件
    把内存中的数据按其在内存中的存储形式原样输出到磁盘上存放。

例如整数10000D在内存中的存储形式以及分别按ASCII码形式和二进制形式输出如下图所示:
ASCII和二进制存储

ASCII文件和二进制文件的比较:
ASCII文件便于对字符进行逐个处理,也便于输出字符;
但一般占存储空间较多,而且要花费转换时间。

二进制文件可以节省外存空间和转换时间;
但一个字节并不对应一个字符,不能直接输出字符形式。

一般中间结果数据需要暂时保存在外存上,以后又需要输入内存的,常用二进制文件保存。

3.C语言对文件的处理方法

缓冲文件系统:
系统自动地在内存区为每一个正在使用的文件开辟一个缓冲区。
用缓冲文件系统进行的输入输出又称为高级磁盘输入输出

非缓冲文件系统:
系统不自动开辟确定大小的缓冲区,而由程序为每个文件设定缓冲区。
用非缓冲文件系统进行的输入输出又称为低级输入输出系统

在UNIX系统下,用缓冲文件系统来处理文本文件,用非缓冲文件系统来处理二进制文件。

ANSI C标准只采用缓冲文件系统来处理文本文件和二进制文件。

C语言中对文件的读写都是用库函数来实现。

二、文件的打开与关闭

1.文件类型指针

定义文件型指针变量:

FILE  *fp;

fp是一个指向FILE类型结构体的指针变量。

我们使fp指向某一个文件的结构体变量,从而通过该结构体变量中的文件信息能够访问该文件。
如果有n个文件,一般应设n个指针变量,使它们分别指向n个文件,以实现对文件的访问。

定义FILE类型的数组:

FILE f[5];

定义了一个结构体数组f,它有5个元素,可以用来存放5个文件的信息。

2.文件的打开(fopen函数)

函数调用:

FILE  *fp;
fp = fopen(文件名, 使用文件方式);

参数说明:
文件名是准备访问的文件的名字;
使用文件方式是读还是写等;
fp指定指向被打开的文件的指针变量。

文件使用方式如下:

方式含义
r(只读)为输入打开一个文本文件
w(只写)为输出打开一个文本文件
a(追加)向文本文件尾增加数据
rb(只读)为输入打开一个二进制文件
wb(只写)为输出打开一个二进制文件
ab(追加)向二进制文件尾增加数据
r+(读写)为读/写打开一个文本文件
w+(读写)为读/写建立一个新的文本文件
a+(读写)为读/写打开一个文本文件
rb+
wb+(读写)为读/写建立一个新的二进制文件
ab+(读写)为读/写打开一个二进制文件

说明:
(1)凡用r方式打开一个文件时,该文件必须已经存在,且只能从该文件读出。

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

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

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

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

文件打开练习如下:

#include <stdio.h>
#include <stdlib.h>

int main(){
	FILE *fp;
	if(!(fp = fopen("E:\\Test\\test.txt", "r"))){
		printf("Can not open the file.\n");	
	}
	else{
		printf("Open success.\n");
	}
	
	return 0;
}

打印:

Open success.

3.文件的关闭(fclose函数)

函数调用:

fclose(文件指针);

函数功能:
使文件指针变量不指向该文件,也就是文件指针变量与文件脱钩,此后不能再通过该指针对原来与其相联系的文件进行读写操作。

返回值:
关闭成功返回值为0,否则返回EOF(-1)。

三、文件的读写

对文件的读和写是最常用的文件操作,在C语言中提供了多种文件读写的函数,使用这些函数时都要包含头文件stdio.h

1.字符读写函数fgetc和fputc

fputc()函数调用:

fputc( ch, fp);

函数功能:
将字符(ch的值)输出到fp所指向的文件中去。

用写或读写方式打开一个已存在的文件时将清除原有的文件内容,写入字符从文件首开始;
如需保留原有文件内容,希望写入的字符以文件末开始存放,必须以追加方式打开文件;
被写入的文件若不存在,则创建该文件。

文件写入练习:

#include <stdio.h>
#include <stdlib.h>

int main(){
	FILE *fp;
	char ch, filename[20];
	printf("Please input the filename ypu want to write: ");
	scanf("%s", filename);
	if(!(fp = fopen(filename, "wt+"))){
		printf("Cannot open the file!\n");
		exit(0);
	}
	printf("Please input the sentences you want to write:\n");
	ch = getchar();
	ch = getchar();
	while(ch != EOF){
		fputc(ch, fp);
		ch = getchar();
	}
	fclose(fp);
		
	return 0;
}

打印:

Please input the filename ypu want to write: test.txt
Please input the sentences you want to write:
I love c!
^Z

此时查看源程序同级目录,可以看到新增加了一个文件test.txt,里面的内容为:

I love c!

每写入一个字符,文件内部位置指针向后移动一个字节。

fputc函数有一个返回值,如写入成功则返回写入的字符,否则返回一个EOF,可用此来判断写入是否成功。

fgetc()函数调用:

ch = fgetc(fp);

函数功能:
其意义是从打开的文件fp中读取一个字符并传入ch中。

在fgetc函数调用中,读取的文件必须是以读或读写方式打开的。

在文件内部有一个位置指针,用来指向文件的当前读写字节。

在文件打开时,该指针总是指向文件的第一个字节;
使用fgetc函数后,该位置指针将向后移动一个字节。
因此可连续多次使用fgetc函数,读取多个字符。

文件指针和文件内部的位置指针不是一回事:
文件指针是指向整个文件的,须在程序中定义说明,只要不重新赋值,文件指针的值是不变的;
文件内部的位置指针用以指示文件内部的当前读写位置,每读写一次,该指针均向后移动,它不需在程序中定义说明,而是由系统自动设置的。

文件读出练习:

#include <stdio.h>
#include <stdlib.h>

int main(){
	FILE *fp;
	char ch, filename[20];
	printf("Please input the filename ypu want to read: ");
	scanf("%s", filename);
	if(!(fp = fopen(filename, "r"))){
		printf("Cannot open the file!\n");
		exit(0);
	}
	while(ch != EOF){
		ch = fgetc(fp);
		putchar(ch);
	}
	fclose(fp);
		
	return 0;
}

打印:

Please input the filename ypu want to read: test.txt
I love c!

从一个文本文件顺序读入字符并在屏幕上显示出来:

ch = fgetc(fp);
while(ch != EOF){
    putchar(ch);
    ch = fgetc(fp);
}

EOF不是可输出字符,因此不能在屏幕上显示;
由于字符的ASCII码不可能出现-1,因此EOF定义为-1是合适的。
当读入的字符值等于-1时,表示读入的已不是正常的字符而是文件结束符。

从一个二进制文件顺序读入字符:

while(!feof(fp){
	ch = fgetc(fp);
})

ANSI C提供一个feof()函数来判断文件是否真的结束:
如果是文件结束,函数feof(fp)的值为1(真);
否则为0(假)。
这也适用于文本文件的读取。

二进制文件读写练习:
实现图片文件合成器。
代码如下:

#include <stdio.h>
#include <stdlib.h>

int main(){
	FILE *f_pic, *f_file, *f_merged;
	char pic_name[50], file_name[50], merged_name[50], ch;
	printf("请输入要合成的图片和文件的名称:\n");
	printf("Picture: ");
	scanf("%s", pic_name);
	printf("File: ");
	scanf("%s", file_name);
	printf("Result: ");
	scanf("%s", merged_name);
	if(!(f_pic = fopen(pic_name, "rb"))){
		printf("Can not open the picture %s !", pic_name);
		return;
	}
	if(!(f_file = fopen(file_name, "rb"))){
		printf("Can not open the file %s !", file_name);
		return;
	}
	if(!(f_merged = fopen(merged_name, "wb"))){
		printf("Can not open the file %s !", file_name);
		return;
	}
	while(!(feof(f_pic))){
		ch = fgetc(f_pic);
		fputc(ch, f_merged);
	}
	fclose(f_pic);
	while(!(feof(f_file))){
		ch = fgetc(f_file);
		fputc(ch, f_merged);
	}
	fclose(f_file);
	fclose(f_merged);
	
	system("pause");
		
	return 0;
}

打印:

请输入要合成的图片和文件的名称:
Picture: 520表白.jpg
File: 让人无法拒绝的8句表白.pdf
Result: 520.m

操作示例:

小程序实现封装表白

2.字符串读写函数fgets和fputs

fgets函数调用形式如:

fgets(str,n,fp);

函数作用:
从fp所指的文件中读出n-1个字符送入字符数组str中,因为在最后加一个'\0'

返回值:
str的首地址。

fputs函数调用方式:

fputs(str,fp);

函数作用:
其意义是把字符串str写入fp所指的文件之中。

返回值:
输入成功,返回值为0;
输入失败,返回EOF。

字符串读出练习:

#include <stdio.h>
#include <stdlib.h>

#define LEN 21

int main(){
	FILE *fp;
	char buffer[LEN];
	if(!(fp = fopen("test.txt", "rt"))){
		printf("\nCannot open file");
		return;
	}
	fgets(buffer, LEN, fp);
	printf("%s\n", buffer);
	fclose(fp);
		
	return 0;
}

打印:

I love c!


字符串写入练习:

#include <stdio.h>
#include <stdlib.h>

#define LEN 20

int main(){
	FILE *fp;
	char ch, buffer[LEN];
	if(!(fp = fopen("test.txt", "at+"))){
		printf("\nCannot open file");
		return;
	}
	printf("Please input a string:\n");
	fgets(buffer, LEN, stdin);
	fputs(buffer, fp);
	rewind(fp);
	printf("\nTHe file is:\n");
	ch = fgetc(fp);
	while(ch != EOF){
		putchar(ch);
		ch = fgetc(fp);
	}
	printf("\n");
	fclose(fp);
		
	return 0;
}

打印:

Please input a string:
I'm Corley!!

THe file is:
I love c!
I'm Corley!!


这里使用fgets(buffer, LEN, stdin)而不是scanf("%s", buffer)是因为scanf()函数默认遇到空格或回车键就会判断为输入结束,如果字符串中有空格或换行符就会导致输入不完整;
使用rewind();是将文件内部指针移到文件开头,便于后边从头读出文件。

3.数据块读写函数fread和fwrite

数据块读写函数函数调用:

fread(buffer, size, count, fp);
fwrite(buffer, size, count, fp); 

参数说明:

  • buffer
    是一个指针:
    对fread 来说,它是读入数据的存放地址;
    对fwrite来说,是要输出数据的地址(均指起始地址);
  • size
    要读写的字节数。
  • count
    要进行读写的数据项的字节数。
  • fp
    文件型指针。

若有如下结构类型:

struct student_type{
	char name[10];
    int num;
    int age;
    char addr[30];} stud[40];

可以用fread和fwrite来进行数据的操作:

for(i = 0; i < 40; i++>{
	fread(&stud[i], sizeof(struct student_type), 1, fp);
})
for(i = 0; i < 40; i++>{
	fwrite(&stud[i], sizeof(struct student_type), 1, fp);
})

练习:
从键盘输入4个学生的有关数据,然后把它们以二进制的格式存储到磁盘文件中。
代码如下:

#include <stdio.h>
#include <stdlib.h>

#define SIZE 4

struct student{
	char name[10];
	int num;
	int age;
	char addr[20];
} stu[SIZE];

int main(){
	void save();
	int i;
	printf("Please input the student's name, num, age and address:\n");
	for(i = 0; i < SIZE; i++){
		scanf("%s %d %d %s", &stu[i].name, &stu[i].num, &stu[i].age, &stu[i].addr);
	}
	save();
		
	return 0;
}

void save(){
	FILE *fp;
	int i;
	if(!(fp = fopen("student_list", "wb"))){
		printf("\nCannot open file");
		return;
	}
	
	for(i=0;i < SIZE; i++){
		if(fwrite(&stu[i], sizeof(struct student), 1, fp) != 1){
			printf("File write error!\n");
			fclose(fp);
		}
	}
}

打印:

Please input the student's name, num, age and address:
Corley 101 18 Road1
Jack 102 20 Road2
Shirley 103 19 ROad3
Tom 104 17 Road4

此时发现在与源程序同级的目录下多了一个文件student_list。

数据块读出练习:

#include <stdio.h>
#include <stdlib.h>

#define SIZE 4

struct student{
	char name[10];
	int num;
	int age;
	char addr[20];
} stu[SIZE];

int main(){
	void load();
	int i;
	load();
	printf("      name   num   age    address\n");
	for(i = 0; i < SIZE; i++){
		printf("%10s %5d %5d %10s\n", stu[i].name, stu[i].num, stu[i].age, stu[i].addr);
	}
		
	return 0;
}

void load(){
	FILE *fp;
	int i;
	if(!(fp = fopen("student_list", "rb"))){
		printf("Cannot open file\n");
		return;
	}
	
	for(i = 0;i < SIZE; i++){
		fread(&stu[i], sizeof(struct student), 1, fp);
	}
	
	fclose(fp);
}

打印:

      name   num   age    address
    Corley   101    18      Road1
      Jack   102    20      Road2
   Shirley   103    19      ROad3
       Tom   104    17      Road4

4.格式化读写函数fprintf和fscanf

函数调用:

fprintf(文件指针, 格式字符串, 输出表列);
fscanf(文件指针, 格式字符串, 输入表列); 

函数功能:
从磁盘文件中按格式读入或输出字符。

例如:

fprintf(fp, "%d, %6.2f", i, t);
fscanf(fp, "%d, %f", &i, &t);

5.顺序读写和随机读写

顺序读写:
位置指针按字节位置顺序移动。

随机读写:
读写完上一个字符(字节)后,并不一定要读写其后续的字符(字节),而可以读些文件中任意位置上所需要的字符(字节)。

四、文件的定位

1.fseek函数

一般用于二进制文件。

函数调用形式:

fseek(文件类型指针, 位移量, 起始点);

函数功能:
改变文件的位置指针。

参数说明:
起始点:

文件位置标识数字
文件开头SEEK_SET0
文件当前位置SEEK_CUR1
文件末尾SEEK_END2

位移量:
以起始点为基点,向前移动的字节数。

使用举例:
fseek(fp, 100L, 0);将位置指针移到离文件头100个字节处;
fseek(fp, 50L, 1);将位置指针移到离当前位置50个字节处;
fseek(fp, 50L, 2);将位置指针从文件末尾处向后退10个字节;
fseek(fp, i*sizeof(struct stu), 0);将位置指针移到离文件头i*sizeof(struct stu)距离处。

练习:
从前面的学生文件student_list中读出第2个学生的数据。
代码如下:

#include <stdio.h>
#include <stdlib.h>

#define SIZE 4

struct student{
	char name[10];
	int num;
	int age;
	char addr[20];
} stu;

int main(){
	FILE *fp;
	int i = 1;
	if(!(fp = fopen("student_list", "rb"))){
		printf("Cannot open file\n");
		return;
	}
	rewind(fp);
	fseek(fp, i*sizeof(struct student), 0);
	fread(&stu, sizeof(struct student), 1, fp);
	printf("      name   num   age    address\n");
	printf("%10s %5d %5d %10s\n", stu.name, stu.num, stu.age, stu.addr);
		
	return 0;
}

打印;

      name   num   age    address
      Jack   102    20      Road2

2.ftell函数

函数作用:
得到流式文件中的当前位置,用相对于文件开头的位移量来表示。

返回值:
返回当前位置,出错时返回-1L。

应用举例:

i = ftell(fp);
if(i == -1L){
	printf(“error\n”);
}	

五、出错的检测

1.ferror函数

调用形式:

ferror(fp);

返回值:
返回0,表示未出错;
返回非0,表示出错。

注意:
在调用一个输入输出函数后立即检查ferror函数的值,否则信息会丢失。
在执行fopen函数时,ferror函数的初始值自动置为0。

2.clearerr函数

调用形式:

clearerr(fp);

函数作用:
使文件错误标志和文件结束标志置为0。

只要出现错误标志,就一直保留,直到对同一文件调用clearerr函数或rewind函数,或任何其他一个输入输出函数。

六、文件操作小结

1.文件操作

分类函数名功能
打开文件fopen()打开文件
关闭文件fclose()关闭文件
文件定位fseek()改变文件位置指针的位置
rewind()使文件位置指针重新至于文件开头
ftell()返回文件位置指针的当前值
文件状态feof()若到文件末尾,函数值为真
ferror()若对文件操作出错,函数值为真
clearerr()使ferror()feof()函数值置零

2.文件读写

函数名功能
fgetc(), getc()从指定文件取得一个字符
fputc(), putc()把字符输出到指定文件
fgets()从指定文件读取字符串
fputs()把字符串输出到指定文件
getw()从指定文件读取一个字(int型)
putw()把一个字输出到指定文件
fread()从指定文件中读取数据项
fwrite()把数据项写到指定文件中
fscanf()从指定文件按格式输入数据
fprintf()按指定格式将数据写到指定文件中
  • 13
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 10
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 10
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

AI码东道主

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值