C++(week3):C语言文件操作

(十二) 文件

1.流

在这里插入图片描述


(1)流模型

data sink:数据接收端、数据汇

在这里插入图片描述

优点:
①程序员读写文件时,不需要关心文件的位置
②数据源(data source) 和 数据汇(data sink) 是解耦的


(2)程序员视角的文件

存放的是一个一个字节。
EOF(end of file)指向文件末尾后一个位置,EOF是一个宏,值为-1

在这里插入图片描述


(3)缓冲区类型

①满缓冲:缓冲区空才从输入流读数据;缓冲区满向输出流中写入数据。
行缓冲:以行为单位进行读和写,如stdin、stdout
③无缓冲:没有缓冲区,立即输入输出,例如:标准错误流 stderr

在这里插入图片描述

在这里插入图片描述

刷新输出缓冲区 (fflush),将输出缓冲区的内容输出到屏幕上


(4)标准流

①stdin:标准输入
②stdout:标准输出
③stderr:标准错误

这三个标准流,不需要程序员手动声明、创建、关闭

在这里插入图片描述


(5)二进制文件 与 文本文件

1.区别
(1)二进制文件:byte。(二进制文件以字节为单位,人类不可读,但体积小)

(2)文本文件:字符 + 编码。(文本文件以字符为单位,一个字符占几个字节)

在这里插入图片描述


2.优缺点:
文本文件:人类可读,数据量大
二进制文件:人类不可读,数据量小

在这里插入图片描述


(6)文件流的接口(API)

1.打开文件流 :fopen

2.读/写文件:
统计、转换、加密解密

2.5 移动文件位置

3.关闭文件流:fclose



2.打开/关闭文件

(1)fopen

1.fopen的参数
①filename是文件的路径
②mode是打开模式

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

在这里插入图片描述


2.文件路径
(1)绝对路径
从根目录 (或者盘符) 开始,一直到文件所在的位置,比如:“c:/project/test.dat”。
(2)相对路径
另一种是相对路径:从当前工作目录开始,一直到文件所在的位置,比如:“in.dat”。

相对路径用的多,因为一个app各文件的相对路径一般不变,但是绝对路径,当安装到不同电脑上时一般不同。


3.打开模式(mode):文件的类型、对文件的操作(r,w)
(1)以文本文件方式打开
①“r”,读(read):要求文件存在。若不存在则返回NULL
②“w”,写(write):若文件存在,清空文件内容;若文件不存在,创建文件。
③“a”,追加(append):若文件存在,不修改原内容,每次都在文件末尾追加写入;若文件不存在,创建文件。

权限文件存在文件不存在
r只读打开失败,返回NULL
w清空文件内容创建文件
a追加创建文件
r+读写打开失败,返回NULL
w+读写清空文件内容创建文件
a+追加创建文件

在这里插入图片描述

在这里插入图片描述


(2)以二进制文件打开

在这里插入图片描述


(2)fclose

fclose 可以关闭程序不再使用的文件。

int fclose(FILE* stream);

如果成功关闭, fclose 返回零;否则返回 EOF


(3)示例代码
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>

int main(void) {
	//1.打开文件
	FILE* stream = fopen("a.txt", "w");
	if (stream == NULL) {
		fprintf(stderr, "file open failed.\n");
		exit(1);
	}

	//2.读写文件 (统计,转换,加密,解密)


	//3.关闭文件
	fclose(stream);

	return 0;
}



3.读/写文件

在这里插入图片描述

(1)fgetc / fputc:读写文本文件,一个字符一个字符地读写

1.函数参数

int fgetc(FILE* stream);
int fputc(int c, FILE* stream);

2.惯用法:一个字符一个字符地读取,直到文件末尾

int c;
while((c = fgetc(src)) != EOF){
	//操作
}

3.完整代码:

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <ctype.h>

int main(int argc, char* argv[]) {
	//xxx.exe src dst
	//0.参数校验
	if (argc != 3) {
		fprintf(stderr, "Usage: %s src dst\n", argv[0]);
		exit(1);
	}
	//1.打开文件
	FILE* src = fopen(argv[1], "r");
	if (!src) {
		fprintf(stderr, "Open %s failed.\n", argv[1]);
		exit(1);
	}

	FILE* dst = fopen(argv[2], "w");
	if (!dst) {
		fprintf(stderr, "Open %s failed.\n", argv[2]);
		fclose(src);
		exit(1);
	}

	//2.读写文件 (统计,转换,加密,解密)
	//(1)一个字符一个字符地读写:fgetc, fputc
	//把大写字母转换为小写字母
	int c;
	while ((c = fgetc(src)) != EOF) {
		fputc(tolower(c), dst);
	}

	printf("大写字母已全部转换为小写字母\n");

	//3.关闭文件
	fclose(src);
	fclose(dst);

	return 0;
}

(2)fgets / fputs:一行一行地读写

fgetc的c是character,fgets的s是string

①fgets

1.函数参数

char* fgets(char* str, int count, FILE* stream);
while(fgets(sendline, MAXLINE, stdin) != NULL){ //一行一行地读

}

参数:
①str:指向第一个字符数组
②count:能够写入的最大字符数量 (通常是str指向字符数组的长度)
③stream:输入流
④返回值:成功返回str,失败返回NULL

2.fgets的特点
①fgets会读\n
②gets(str) 等价于 fgets(str, ∞, stdin) ,即gets不会检查数组越界


②fputs

1.函数参数

int fputs(const char* str, FILE* stream);

参数:
①str:要写的字符串(以’\0’结尾的字符串)
②stream:输出流
③返回值:成功返回一个非负值;失败返回EOF,并设置errno


2.fputs的特点
①fputs原样输出字符串,puts在字符串后多输出一个换行符’\n’
②puts(str) 等价于 fputs(str, stdout)

在这里插入图片描述


③代码

1.核心代码:

#define MAXLINE 128
//2.读写文件:
//(2)一行一行地读写
//每行加序号
char line[MAXLINE];
char buffer[MAXLINE];
//fgets(line, MAXLINE, src);
//fgets(line, MAXLINE, src);
int line_num = 1;
while ((fgets(buffer, MAXLINE, src)) != NULL) {
	sprintf(line, "%d.%s", line_num, buffer);
	fputs(line, dst);
	line_num++;
}

2.完整代码:

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <ctype.h>
#define MAXLINE 128

int main(int argc, char* argv[]) {
	//xxx.exe src dst
	//0.参数校验
	if (argc != 3) {
		fprintf(stderr, "Usage: %s src dst\n", argv[0]);
		exit(1);
	}
	//1.打开文件
	FILE* src = fopen(argv[1], "r");
	if (!src) {
		fprintf(stderr, "Open %s failed.\n", argv[1]);
		exit(1);
	}

	FILE* dst = fopen(argv[2], "w");
	if (!dst) {
		fprintf(stderr, "Open %s failed.\n", argv[2]);
		fclose(src);
		exit(1);
	}

	//2.读写文件:
	//(2)一行一行地读写
	//每行加序号
	char line[MAXLINE];
	char buffer[MAXLINE];
	//fgets(line, MAXLINE, src);
	//fgets(line, MAXLINE, src);
	int line_num = 1;
	while ((fgets(buffer, MAXLINE, src)) != NULL) {
		sprintf(line, "%d.%s", line_num, buffer);
		fputs(line, dst);
		line_num++;
	}
	printf("每行内容前面已加上序号。\n");

	//3.关闭文件
	fclose(src);
	fclose(dst);

	return 0;
}

(3)fscanf / fprintf:格式化地读写
①fscanf
int fscanf(FILE* stream, const char* format, ...);

在这里插入图片描述


②fprintf
int fprintf(FILE* stream, const char* format, ...);

在这里插入图片描述


③代码
#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <ctype.h>
#define MAXLINE 128

typedef struct Student {
	int id;
	char name[25];
	char gender;
	int chinese;
	int math;
	int english;
} Student;

int main(int argc, char* argv[]) {
	//xxx.exe src dst
	//0.参数校验
	if (argc != 3) {
		fprintf(stderr, "Usage: %s src dst\n", argv[0]);
		exit(1);
	}
	//1.打开文件
	FILE* src = fopen(argv[1], "r");
	if (!src) {
		fprintf(stderr, "Open %s failed.\n", argv[1]);
		exit(1);
	}

	FILE* dst = fopen(argv[2], "w");
	if (!dst) {
		fprintf(stderr, "Open %s failed.\n", argv[2]);
		fclose(src);
		exit(1);
	}

	//2.读写文件 (统计,转换,加密,解密)
	//(3)格式化地读写
	Student students[5];
	for (int i = 0; i < 5; i++) {
		fscanf(src, "%d%s %c%d%d%d",
			&students[i].id,
			students[i].name,
			&students[i].gender,
			&students[i].chinese,
			&students[i].math,
			&students[i].english);
	}

	//修改成绩
	for (int i = 0; i < 5; i++) {
		students[i].chinese *= 10;
		students[i].math *= 10;
		students[i].english *= 10;
	}

	for (int i = 0; i < 5; i++) {
		fprintf(dst, "%d %s %c %d %d %d\n",
			students[i].id,
			students[i].name,
			students[i].gender,
			students[i].chinese,
			students[i].math,
			students[i].english);
	}

	printf("成绩已修改。\n");

	//3.关闭文件
	fclose(src);
	fclose(dst);

	return 0;
}
//a.txt
1 Edward M 100 100 100
2 Amber  F  99  99  99
3 Sam    M  98  98  98 
4 Windy  F  97  97  97
5 Chole  F  96  96  96

(4)fread / fwrite:读写二进制文件,一块一块地读写
①fread

在这里插入图片描述


②fwrite

在这里插入图片描述


③代码

1.核心代码:

//读写二进制文件 (复制)
char buffer[BUFSIZE];
int bytes;
while ((bytes = fread(buffer, 1, BUFSIZE, src)) > 0) {
	fwrite(buffer, 1, BUFSIZE, dst);
}

2.完整代码:复制二进制文件

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>

#define BUFSIZE 4096

int main(int argc, char* argv[]) {
	//xxx.exe src
	//0.参数校验
	if (argc != 3) {
		fprintf(stderr, "Usage: %s src dst\n", argv[0]);
		exit(1);
	}
	//1.打开文件
	FILE* src = fopen(argv[1], "rb");
	if (!src) {
		fprintf(stderr, "Open %s failed.\n", argv[1]);
		exit(1);
	}

	FILE* dst = fopen(argv[2], "wb");
	if (!dst) {
		fprintf(stderr, "Open %s failed.\n", argv[2]);
		fclose(src);
		exit(1);
	}

	//2.读写文件 
	//(4)读写二进制文件 (复制)
	char buffer[BUFSIZE];
	int bytes;
	while ((bytes = fread(buffer, 1, BUFSIZE, src)) > 0) {
		fwrite(buffer, 1, BUFSIZE, dst);
	}

	//3.关闭文件
	fclose(src);
	fclose(dst);

	return 0;
}



4.文件定位、移动文件位置

(1)fseek:移动文件位置
int fseek(FILE* stream, long int offset, int whence);

whence的三个宏:
SEEK_SET:文件的起始位置,0
SEEK_CUR:文件的当前位置,pos
SEEK_END:文件的末尾位置,EOF

在这里插入图片描述

在这里插入图片描述


(2)ftell:返回当前文件位置
long int ftell(FILE* stream);

在这里插入图片描述


(3)rewind:将文件位置移回开头
void rewind(FILE* stream);

在这里插入图片描述


(4)代码实战

在这里插入图片描述

难点:
①如何确定文件的大小

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

char* readFile(const char* path) {
	//1.打开文件
	FILE* stream = fopen(path, "rb");
	if (stream == NULL) {  //!stream
		fprintf(stderr, "Open %s failed\n", path);
		exit(1);
	}

	//2.确定文件大小
	fseek(stream, 0, SEEK_END);
	long n = ftell(stream);
	char* content = malloc((n + 1) * sizeof(char));  // 1 for '\0'

	//3.读取文件
	rewind(stream);  //回到文件开头
	int bytes = fread(content, 1, n, stream);
	content[bytes] = '\0';

	//4.关闭文件
	fclose(stream);

	return content;
}

int main(int argc, char* argv[]) {
	//0.参数校验
	if (argc != 2) {
		fprintf(stderr, "Usage: %s filename\n", argv[0]);
		exit(1);
	}

	char* content = readFile(argv[1]);

	//操作
	printf("%s\n", content);

	free(content);

	return 0;
}



5.错误处理:perror

1.如何检测错误:①②
2.如何打印错误信息:③④


①返回值


errno
定义在 <errno.h> 头文件中
(每个线程都有自己的errno,线程特定变量)
errno为0表示没有错误,非0表示有错误

#include <errno.h>

FILE* fp = fopen("not_exist.txt", "r");
printf("%d\n", errno); 

strerror(errno)
定义在 <string.h> 头文件中

#include <error.h>
#include <string.h>

FILE* fp = fopen("not_exist.txt", "r");
printf("%s\n", strerror(errno));  //将errno对应的错误信息,转化为人类可读的字符串

perror("前缀信息"),错误的前缀信息,后面自动添加了 : \n

相当于调用了fprintf(stderr, "%s: %s\n", "prefix", strerror(errno));

#include <error.h>
#include <string.h>

FILE* fp = fopen("not_exist.txt", "r");
fprintf(stderr, "%s: %s\n", "prefix", strerror(errno));
perror("prefix");  //添加错误的前缀信息
exit(1);

#define _CRT_SECURE_NO_WARNINGS
#include <stdio.h>
#include <errno.h>
#include <string.h>

int main(void) {
	//errno==0 代表 没有错误
	printf("errno = %d\n", errno);

	FILE* fp = fopen("a.txt","r+"); //实际a.txt不存在
	//errno
	printf("errno = %d\n", errno);
	
	//strerror(errno)
	printf("strerror(errno) = %s\n",strerror(errno));
	
	//perror
	fprintf(stderr, "%s: %s\n", "prefix", strerror(errno));
	perror("prefix");

	return 0;
}

在这里插入图片描述


error()
仅linux可用,自动添加‘\n’

#include <error.h>
#include <errno.h>

error(0, error, "prefix");   //不退出程序,设置errno
error(1, 0, "prefix");      //相当于exit(1),不设置errno
error(1, errno, "prefix");  //相当于exit(1),设置errno

输出:

./可执行程序名: prefix   
  • 15
    点赞
  • 21
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

程序员爱德华

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

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

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

打赏作者

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

抵扣说明:

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

余额充值