C++学习记录——삼십이 C++IO流


1、C++标准IO流

C语言的printf和scanf无法很好的输入输出自定义类型,且还需要程序员自己确定类型,所以C++就引入了输入流和输出流,是设备和内存之间的沟通。

在这里插入图片描述

其实iostream是通过菱形继承实现的。cout,cerr,clog其实做得有时候没有区分

int main()
{
	cout << "1111" << endl;
	cerr << "2222" << endl;
	return 0;
}

也都能打印出来。fstream和sstream是针对文件和字符串的,对应的就是fprintf/fscanf,fwrite/fread,sprintf/sscanf。C++IO流的特点就是面向对象以及能更好地支持自定义类对象的IO。

istream和ostream是防止拷贝的。

C++IO流的缓冲区和C语言的缓冲区会进行同步,比如用cout和printf混合着输出,也能输出。默认开着同步,也可以关闭同步增强效率。和这个接口有关

在这里插入图片描述

实现了迭代器的容器,通常就不需要重载IO流。

	string str;
	while (cin >> str)
	{
		cout << str << endl;
	}

这样的写法,就可以持续写入字符串,想要停止,就用ctrl + z + 换行来停止,这样是将流对象提取到结束,如果是ctrl + c就是信号强杀进程。我们要知道此时cin >> str是在做什么,>>右边的数据会被获取,给到cin,如果有多个,那就获取多个,最终都流向cin,然后返回一个istream类型的对象,这个对象会被转为bool类型来进行判断,但是一些自定义类型不可以转,像string,而istream可以转是因为类里重载了一个bool()函数,支持自定义类型转内置类型。

在之前的博客中,写到过单参数构造的类支持隐式类型转换

class A
{
public:
	A(int a)
		:_a(a)
	{}
private:
	int _a;
};

int main()
{
	A aa1 = 1;
	return 0;
}

这里的内部实现其实是构造一个临时对象,然后用这个对象来拷贝构造aa1,如果是A&就不行了。这是内置类型转自定义类型,但是自定义转内置不可:int x = aa1。解决办法就是重载一个类型()函数。

	operator int()
	{
		return _a;
	}

如果该函数前面加上explicit就不行了,不过在下面写int x = (int)aa1,强制转换一下也行。

istream里的operator bool(),如果被设置成错误的信息,就返回false,否则返回true,错误的信息例如ctrl + Z + 换行。

2、C++文件IO流

1、二进制读写

#include <fstream>
using namespace std; 

int main()
{
	ofstream ofs("F://test.txt");
	ofs << "hello world";
	return 0;
}

默认是覆盖写,第二个模板参数默认是ios_base::openmode mode = ios_base::out,ios_base是ofstream和ifstream的父类,如果想要用其它写入模式

在这里插入图片描述

是用 | 来隔开的。

int main()
{
	ofstream ofs("F://test.txt", ofstream::out | ofstream::app);
	ofs << "hello world";
	return 0;
}

看一个读写都有的代码

struct ServerInfo
{
	char _address[32];
	int _port;
	Date _date;
};

struct ConfigManager
{
public:
	ConfigManager(const char* filename)
		:_filename(filename)
	{}

	void WriteBin(const ServerInfo& info)
	{
		ofstream ofs(_filename, ofstream::out | ofstream::binary);
		ofs.write((const char*)&info, sizeof(info));
	}

	void ReadBin(ServerInfo& info)
	{
		ifstream ifs(_filename, ofstream::in | ofstream:::binary);
		ifs.read((char*)&info, sizeof(info));
	}

private:
	string _filename; // 配置文件
};

int main()
{
	ServerInfo winfo = { "192.0.0.1", 80, {2023, 9, 4} };
	string str;
	cin >> str;
	if (str == "二进制写")
	{
		ConfigManager cm("F://test.txt");
		cm.WriteBin(winfo);
	}
	else if (str == "二进制读")
	{
		ServerInfo rinfo;
		ConfigManager cm("F://test.txt");
		cm.ReadBin(rinfo);
		cout << rinfo._address << endl;
		cout << rinfo._port << endl;
		cout << rinfo._date << endl;
	}
	return 0;
}

二进制读写对象中不能用string,能读进去,但是会崩,而且如果输入的字符串比较长那基本上就是崩掉。

二进制写时,写到文件的是string对象及里面指向空间的指针,程序结束,string自动析构,指针指向的空间销毁了,所以就没法读取,读到的就是野指针。

二进制读写的优点就是简单易操作,缺点就是没法看到实体。

2、文本读写

C语言文本读写的本质是内存中任何类型都转成字符串来交互,C++封装运算符重载后就变得更高效了。

	void WriteText(const ServerInfo& info)
	{
		ofstream ofs(_filename);
		ofs << info._address << " " << info._port << " " << info._date;
	}

日期类中实现了对流插入流提取的重载,不仅ostream能使用,ofstream和osstream都可以用,因为使用了继承体系,ostream是它们的父类。istream也是如此。

	void WriteText(const ServerInfo& info)
	{
		ofstream ofs(_filename);
		ofs << info._address << endl;
		ofs << info._port << endl;
		ofs << info._date << endl;
	}
	void ReadText(ServerInfo& info)
	{
		ifstream ifs(_filename);
		ifs >> info._address;
		ifs >> info._port;
		ifs >> info._date;
	}
	
	else if (str == "文本写")
	{
		ConfigManager cm("F://test.txt");
		cm.WriteText(winfo);
	}
	else if (str == "文本读")
	{
		ServerInfo rinfo;
		ConfigManager cm("F://test.txt");
		cm.ReadText(rinfo);
		cout << rinfo._address << endl;
		cout << rinfo._port << endl;
		cout << rinfo._date << endl;
	}

3、stringstream

对应C语言的sprintf和sscanf。

在这里插入图片描述

就像最上面那个图中显示,ios_base是一个基类,ios是它的派生类,istream和ostream是ios的派生类,iostream是istream和ostream的组合,然后istream,ostream,iostream各有分支,stingstream继承于iostream,istringstream继承于istream,ostringstream继承于ostream。

头文件是sstream,不是stringstream。

int main()
{
	ostringstream oss;
	oss << 100 << " ";
	oss << 11.22 << " ";
	oss << "hello  wolrd" << endl;
	string str = oss.str();
	cout << str << endl;
	return 0;
}

可以把流插入的内容转换成一个字符串,也可以提取出来。

int main()
{
	ostringstream oss;
	oss << 100 << " ";
	oss << 11.22 << " ";
	oss << "hello  wolrd" << endl;
	string str = oss.str();
	cout << str << endl;

	istringstream iss(str);
	int i;
	double d;
	string s;
	iss >> i >> d >> s;
	cout << i << endl;
	cout << d << endl;
	cout << s << endl;
	return 0;
}

iss后面的顺序也得和oss拿到的顺序一样,要不然会使得某些变量内容不对,但也能打印出来。oss和iss也可以换成stringstream的类型。

这个功能主要用于序列化和反序列化

struct ChatInfo
{
	string _name;
	int _id;
	string _msg;
};

int main()
{
	ChatInfo winfo = { "张三", 123456, "你好" };
	return 0;
}

像这样,我们就可以用ostringstream把内容放到一个字符串,然后发送给需要这些信息的人,别人收到后,就可以用istringstream来提取。不过插入的时候每个后面都加上换行或者空格,因为提取时默认以换行或者空格来分割。

复杂的实际情况还有其它方法。

C++IO流

结束。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值