C++系列-文件操作

文件操作

文件类型

  • 文件的读写方式有两种,文本文件和二进制文件。
  • 计算机的存储在物理上是二进制的,所以文本文件与二进制文件的区别并不是物理上的,而是逻辑上的。

文本文件

  • 文本文件是基于字符编码的文件,常见的编码有ASCII编码,UNICODE编码等等。

二进制文件

  • 二进制文件有特定的编码方式,用户一般不能直接读懂。
  • 二进制文件是基于值编码的文件,你可以根据具体应用,指定某个值是什么意思(这样一个过程,可以看作是自定义编码。
  • 二进制文件可看成是变长编码的,因为是值编码嘛,多少个比特代表一个值,完全由你决定。大家可能对BMP文件比较熟悉,就拿它举例子吧,其头部是较为固定长度的文件头信息,前2字节用来记录文件为BMP格式,接下来的8个字节用来记录文件长度,再接下来的4字节用来记录bmp文件头的长度。

文件操作的三大类

需要包含头文件,fstream, f:file

  • ofstream, o: out, 代码往外吐数据,即写到文件去。
  • ifstream, i: in, 入向代码,从文件读到代码中。
  • fstream,in,out均包括。

文件的打开方式

打开方式 适用对象 说明
ios::in ifstream,fstream 打开文件用于读取数据,如果文件不存在,则打开出错
ios::out ofstream,fstream 打开文件用于写入数据。如果文件不存在,则新建该文件;如果文件原来就存在,则打开时清除原来的内容
ios::app ofstream,fstream 打开文件,用于在其尾部添加数据。如果文件不存在,则新建该文件,每次的写入操作都会在后面追加
ios::ate ofstream,fstream 打开一个已有的文件,并将文件指针指向文件末尾。如果文件不存在,则打开出错
ios:: trunc ofstream 打开文件时会清空内部存储的所有数据,相当于先删除原有文件,再重新建立一个文件
ios::binary ifstream,ofstream,fstream 以二进制方式打开文件。若不指定此模式,则以文本模式打开
ios::in ios::out | ios::trunc | fstream 打开文件,既可读取其内容,也可向其写入数据。如果文件本来就存在,则打开时清除原来的内容;如果文件不存在,则新建该文件

ios::app(append)和 ios::ate(at end)

  • 每个文件都有一个位置指针,指向将要读写的下一个字节,打开文件时,默认的位置指针指向文件的开始;打开方式ate(表示at-end)和app(表示追加)能将位置指针定位到文件尾。

  • 如果以追加方式打开文件,所有向文件的输出从当前文件的尾部开始进行,不必重新定位。即使你修改文件位置指针不指向文件尾部,也不会将输出写到那里。

    以at-end方式打开文件时,文件起始位置指向文件尾部,但你可以修改位置指针指向其它位置,输出则写到指针指向的位置。

  • 在实际应用中,如果需要在多个线程或进程间安全地追加数据到文件中,而不希望数据被覆盖或混淆,那么ios::app模式是最佳选择。

  • 而如果需要在打开文件时就确定写指针的位置,或者需要清空文件内容然后重新写入,那么使用ios::ate模式可能更为合适, 然而,值得注意的是,如果只使用ios::ate模式而不配合ios::in或ios::out模式,将会清空原文件的内容。

  • 当使用ios::app模式时,如果打开的文件不存在,系统将自动创建该文件。

  • 而当使用ios::ate模式时,如果打开的文件不存在,系统将无法打开文件,报错。

写文件

写文件文件步骤

  • 包含头文件, #include <fstream>
  • 创建流对象,通过流对象,可以与文件打交道,ofstream ofs
  • 打开文件,参数有文件的路径,打开方式等,ofs.open(文件路径,打开方式), 也可以创建流对象和打开合并为一个ofstream ofs(文件路径,打开方式)
  • 向文件写入内容 ofs << 写入的数据
  • 关闭文件, ofs.close()

读文件文件步骤

  • 包含头文件, #include <fstream>
  • 创建流对象,通过流对象,可以与文件打交道,ifstream ofs
  • 打开文件,参数有文件的路径,打开方式等,ofs.open(文件路径,打开方式), 也可以创建流对象和打开合并为一个ifstream ifs(文件路径,打开方式)
  • 从文件读取内容
    -- 方法1,创建字符数组接收读取文件的内容,使用ifs >> 字符数组,while (ifs >> buf)
    -- 方法2,创建字符数组接收读取文件的内容,使用ifs.getline(字符数组,sizeof(字符数组)),(ifs.getline(buf, sizeof(buf)))
    -- 方法3,创建字符串接收读取文件的内容,使用string的全局函数getline(流对象,字符串变量),while (getline(ifs, buf))
    -- 方法4,创建字符接收读取文件的内容,使用ifs.get(),while(char_data = ifs.get())!=EOF)
  • 关闭文件, ofs.close()
cpp 复制代码
#include <fstream>
#include <iostream>
#include <string>
using namespace std;
void main()
{
	ofstream ofs("d:/student.txt", ios::ate);		// 创建流对象,并打开
	ofs << "张三,10" << endl << "李四,20" << endl << "王五,30" << endl;	// 写入文件
	ofs.close();

	ifstream ifs("d:/student.txt", ios::in);		// 创建流对象,写读取方式打开文件
	if (!ifs.is_open())								// 判断文件是否打开成功
		return;
	
	// 方法1,使用ifs >> buf的方式,用字符数组来接收文件中的数据
	//char buf[100] = { 0 };						// 创建字符数组用于存放读取的数据 
	//while (ifs >> buf)							// ifs >> buf, 将文件中的数据依次放入buf中,当读完后,此条件不满足
	//{
	//	cout << buf << endl;
	//}

	// 方法2,使用流对象的方法getline(char* 接收数据的字符数组的地址,sizeof(要接收数据的字符数组))
	//char buf[100] = { 0 };						// 创建字符数组,用于存放读取的数据 
	//while (ifs.getline(buf, sizeof(buf)))			// 使用对象方法getline将文件中的数据放入buf中,当读完后,此条件不满足
	//{
	//	cout << buf << endl;
	//}
	//ifs.close();
	//system("pause");

	// 方法3,使用全局函数getline
	//string buf;
	//while (getline(ifs, buf))		// 使用全局方法getline(输入流对象,接收数据的字符串)#include <string>需要包含,将文件中的数据放入buf中,当读完后,此条件不满足
	//{
	//	cout << buf << endl;
	//}

	// 方法4,逐个字符读取, 这种方式的效率比较低
	char char_data;
	while ((char_data = ifs.get())!=EOF)		// EOF: end of files
	{
		cout << char_data;
	}
}

二进制文件读写

写一般数据

cpp 复制代码
#include <fstream>			// 包含头文件
#include <iostream>
using namespace std;

void main()
{
	char str1[30] = "张三,10\n李四,20";
	char str2[30] = {0};

	ofstream ofs("d:/person.txt", ios::out | ios::binary);	// 创建输出流对象,打开文件,二进制写入方式ios::binary
	if (!ofs)
	{
		cout << "创建文件失败" << endl;
		return;
	}
	ofs.write((const char*)str1, sizeof(str1));      //write以char *的方式进行写出,做一个转化,write(要写入的数据地址,数据长度)
	ofs.close();	// 关闭文件

	
	ifstream ifs("d:/person.txt", ios::in | ios::binary);
	if (!ifs)
	{
		cout << "读取文件失败" << endl;
		return;
	}
	ifs.read((char*)str2, sizeof(str2));		// read(char* 读取数据的接收地址,接收数据的字符数组长度)
	cout << str2 << endl;
	ifs.close();	// 关闭文件
	system("pause");
}

写特殊数据

cpp 复制代码
#include <fstream>
#include <iostream>
using namespace std;
class Person
{
public:
	Person(string name, int age) : m_name(name), m_age(age) {}	// 初始化列表
	string m_name;
	int m_age;
};
void main()
{
	Person p1("张三", 10);
	Person p2("1", 0);
	
	ofstream ofs("d:/person.txt", ios::out | ios::binary);
	ofs.write((const char*)&p1, sizeof(p1));
	ofs.close();

	ifstream ifs("d:/person.txt", ios::in | ios::binary);
	ifs.read((char*)&p2, sizeof(p2));
	if (!ifs)
	{
		cout << "读取文件失败" << endl;
		return;
	}
	cout << p2.m_name << ", " << p2.m_age << endl;
	ifs.close();
	system("pause");
}
相关推荐
起名字真南8 分钟前
【OJ题解】C++实现字符串大数相乘:无BigInteger库的字符串乘积解决方案
开发语言·c++·leetcode
少年负剑去8 分钟前
第十五届蓝桥杯C/C++B组题解——数字接龙
c语言·c++·蓝桥杯
cleveryuoyuo8 分钟前
AVL树的旋转
c++
tyler_download19 分钟前
golang 实现比特币内核:实现基于椭圆曲线的数字签名和验证
开发语言·数据库·golang
小小小~19 分钟前
qt5将程序打包并使用
开发语言·qt
hlsd#20 分钟前
go mod 依赖管理
开发语言·后端·golang
小春学渗透21 分钟前
Day107:代码审计-PHP模型开发篇&MVC层&RCE执行&文件对比法&1day分析&0day验证
开发语言·安全·web安全·php·mvc
杜杜的man24 分钟前
【go从零单排】迭代器(Iterators)
开发语言·算法·golang
亦世凡华、24 分钟前
【启程Golang之旅】从零开始构建可扩展的微服务架构
开发语言·经验分享·后端·golang
神仙别闹31 分钟前
基于MFC实现的赛车游戏
c++·游戏·mfc