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

发布于:2024-05-11 ⋅ 阅读:(121) ⋅ 点赞:(0)

(十二) 文件

1.流

在这里插入图片描述


(1)流模型

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

在这里插入图片描述

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


(2)程序员视角的文件

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

在这里插入图片描述


(3)缓冲区类型

①满缓冲:缓冲区空才从输入流读数据;缓冲区满向输出流中写入数据。
②行缓冲:以行为单位进行读和写
③无缓冲:没有缓冲区,立即输入输出,例如:标准错误流 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 追加 创建文件

在这里插入图片描述

在这里插入图片描述


(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:一行一行地读写
①fgets

1.函数参数

char* fgets(char* str, int count, FILE* stream);

参数:
①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, ...);

在这里插入图片描述


(4)fread / fwrite:读写二进制文件
①fread

在这里插入图片描述


②fwrite

在这里插入图片描述



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

(1)fseek
int fseek(FILE* stream, long int offset, int whence);

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

在这里插入图片描述


(2)ftell
long int ftell(FILE* stream);

在这里插入图片描述


(3)rewind
void rewind(FILE* stream);

在这里插入图片描述


(4)课堂小练习

在这里插入图片描述

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



5.错误处理:perror

1.如何检测错误
2.如何打印错误信息

①返回值
errno,定义在 <errno.h> 头文件中
(每个线程都有自己的errno,线程特定变量)
strerror(errno),定义在 <string.h> 头文件中
perror,错误的前缀信息,后面自带了
相当于调用了fprintf(stderr,"%s: ",%s)


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

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

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

	printf("%s\n", strerror(errno));  //得到errno对应的错误信息,得到人类可读的字符串

	perror("prefix");  //错误的前缀信息

	return 0;
}

网站公告

今日签到

点亮在社区的每一天
去签到