我的编程空间,编程开发者的网络收藏夹
学习永远不晚

C++文件IO流及stringstream流读写文件和字符串操作详解

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

C++文件IO流及stringstream流读写文件和字符串操作详解

一、引入

int main()
{
	string str;
	while (cin >> str)
	{
		cout << str << endl;
	}
	return 0;
}

我们在OJ的时候经常会用到while(cin >> str),这里的流提取实际上是个阻塞操作,只要缓冲区还有数据就继续读,默认以空格或者换行结束,有空格说明是把两段字符串尾插到str。

那么它是怎么结束呢?

答案是输入[Ctrl]-c或者[Ctrl]-z + 换行

[Ctrl]-c是发送信号结束进程。

[Ctrl]-z + 换行是通过返回值条件判断结束while循环,具体看下面讲解。

二、自定义类型隐式类型转换

cin >> str的返回值是一个istream类

实际上返回的就是cin对象。而c++98支持了隐式类型转换,把istream转换为bool,所以能够条件判断。

具体是怎么转换的呢?

看下面这个例子:

class A
{
public:
	A(int a)
		: _a(a)
	{}
private:
	int _a;
};
int main()
{
	// 内置类型转换成自定义类型
	A a = 1;
	return 0;
}

这里按道理来说是构造一个临时对象再拷贝构造,而编译器优化成了直接构造。如果没有单参数的构造函数就无法转换。

那如果我们想要让自定义类型转换成内置类型呢?

直接int aa = a;肯定会报错。

但是我们可以加一个特殊的重载函数。

class A
{
public:
	A(int a)
		: _a(a)
	{}
	operator int()
	{
		return _a;
	}
private:
	int _a;
};
int main()
{
	// 内置类型转换成自定义类型
	A a = 1;
	// 自定义类型转化成内置类型
	int aa = a;
	cout << aa << endl;
	return 0;
}

而我们上面说的把istream转化成bool类型就是类似这样实现的。

operator bool() 里面会检查是特殊字符([Ctrl]-z )就会返回false。

三、sync_with_stdio同步

我们知道cin和scanf都有自己的缓冲区,而如果我们用scanf写入再用cout输出,就会导致速度变慢很多(缓冲区拷贝)。

而sync_with_stdio函数是一个“是否兼容stdio”的开关,C++为了兼容C,保证程序在使用了std::printf和std::cout的时候不发生混乱,将输出流绑到了一起。

决定C++标准streams(cin,cout,cerr…)是否与相应的C标准程序库文件(stdin,stdout,stderr)同步,也就是是否使用相同的stream缓冲区,缺省情况是同步的,但由于同步会带来某些不必要的负担,因此该函数作用就是我们自己可以取消同步 。

#include <iostream>
int main() 
{
    std::ios::sync_with_stdio(false);
    std::cin.tie(0);
    // IO
}

四、文件IO流

文件的读写有两种:

1️⃣ 二进制读写

2️⃣ 文本读写

ofstream是写入文件,而ifstream是从文件中读取。

4.1 open和close文件

这里的参数表示我们想以什么样的方式打开文件。

比方说当我们想用二进制的方式打开文件:

ofs.open ("test.txt", std::ofstream::out | std::ofstream::binary)

而我们也可以在构造的时候直接传进参数。

ofstream ofs("test.txt", std::ios_base::out | std::ios_base::binary)

4.2 写入文件与读出文件

struct ServerInfo
{
	char _address[32];
	int _port;
};
struct Config
{
public:
	Config(const char* filename)
		: _filename(filename)
	{}
	void Write(ServerInfo info)
	{
		ofstream ofs("test.txt", std::ios_base::out | std::ios_base::binary);
		ofs.write((char*)&info, sizeof info);
	}
	void Read(ServerInfo& info)
	{
		ifstream ifs("test.txt", std::ios_base::in | std::ios_base::binary);
		ifs.read((char*)&info, sizeof info);
	}
private:
	string _filename;
};
int main()
{
	Config con("text.txt");
	ServerInfo si = { "aaaaaa", 910 };
	con.Write(si);
	return 0;
}

而我们也可以把数据读回来。

int main()
{
	Config con("text.txt");
	//ServerInfo si = { "aaaaaa", 910 };
	//con.Write(si);
	ServerInfo si;
	con.Read(si);
	cout << si._address << " " << si._port << endl;
	return 0;
}

可以看到内存中和写出去显示出来的不一样。

当然我们可以用文本读写的方式。

struct ServerInfo
{
	char _address[32];
	int _port;
};
struct Config
{
public:
	Config(const char* filename)
		: _filename(filename)
	{}
	void Write(ServerInfo info)
	{
		ofstream ofs(_filename);
		// 重载
		ofs << info._address << endl;
		ofs << info._port << endl;
	}
	void Read(ServerInfo& info)
	{
		ifstream ifs(_filename);
		// 重载
		ifs >> info._address;
		ifs >> info._port;
	}
private:
	string _filename;
};
int main()
{
	Config con("text.txt");
	ServerInfo si = { "aaaaaa", 910 };
	con.Write(si);
	
	return 0;
}

五、stringstream流的使用

在程序中如果想要使用stringstream,必须要包含头文件。在该头文件下,标准库三个类:

istringstream、ostringstream 和 stringstream,分别用来进行流的输入、输出和输入输出操作。

5.1 将数值类型数据格式化为字符串

int main()
{
	int a = 123;
	const char* b = "456";
	double c = 78.9;
	ostringstream os;
	os << a;
	os << b;
	os << c;
	cout << os.str() << endl;
	return 0;
}

当然我们也可以把每个数据都提取出来。但此时输入的时候就要空格或者换行隔开。

int main()
{
	int a = 123;
	const char* b = "456";
	double c = 78.9;
	ostringstream os;
	os << a << " ";
	os << b << " ";
	os << c << " ";
	string ret = os.str();
	cout << ret << endl;
	int d;
	char e[20];
	double f;
	istringstream is(ret);
	is >> d >> e >> f;
	cout << d << " ";
	cout << e << " ";
	cout << e << " ";
	return 0;
}

5.2 序列化和反序列化

序列化指的是将一个内存对象转化成一串字节数据(存储在一个字节数组中),可用于保存到本地文件或网络传输。反序列化就是将字节数据还原成内存对象。

总结

序列化:将对象变成字节流的形式传出去。

反序列化:从字节流恢复成原来的对象。

简单来说,对象序列化通经常使用于两个目的:

1️⃣ 将对象存储于硬盘上,便于以后反序列化使用;

2️⃣ 在网络上传送对象的字节序列

我们现在模拟一个聊天的发送窗口。

class Date
{
	friend ostream& operator << (ostream& out, const Date& d);
	friend istream& operator >> (istream& in, Date& d);
public:
	Date(int year = 1, int month = 1, int day = 1)
		:_year(year)
		, _month(month)
		, _day(day)
	{}
private:
	int _year;
	int _month;
	int _day;
};
istream& operator >> (istream& in, Date& d)
{
	in >> d._year >> d._month >> d._day;
	return in;
}
ostream& operator << (ostream& out, const Date& d)
{
	out << d._year << " " << d._month << " " << d._day;
	return out;
}
struct ServerInfo
{
	friend istream& operator >> (istream& in, ServerInfo& si);
	friend ostream& operator << (ostream& out, ServerInfo& si);
	string _name;// 昵称
	Date _d;// 时间
	string _msg;// 信息
};
istream& operator >> (istream& in, ServerInfo& si)
{
	in >> si._name  >> si._d >> si._msg;
	return in;
}
ostream& operator << (ostream& out, ServerInfo& si)
{
	out << si._name << " ";
	out << si._d << " ";
	out << si._msg << " ";
	return out;
}
int main()
{
	ServerInfo p{ "海阔天空", {2023, 4, 19}, "hello" };
	stringstream os;
	os << p;
	string ret = os.str();
	ServerInfo is;
	stringstream oss(ret);
	oss >> is;
	cout << "-------------------------------------------------------" << endl;
	cout << "昵称:" << is._name << " ";
	cout << is._d << endl;
	cout << is._name << ": " << is._msg << endl;
	cout << "-------------------------------------------------------" << endl;
	return 0;
}

到此这篇关于C++文件IO流及stringstream流读写文件和字符串操作详解的文章就介绍到这了,更多相关C++文件IO流内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

C++文件IO流及stringstream流读写文件和字符串操作详解

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

C++文件IO流及stringstream流读写文件和字符串操作详解

本文详细介绍C++中的文件IO流和stringstream流的使用方法,包括文件的打开、读写操作,以及字符串的输入输出、转换等操作。同时提供实用的示例代码和技巧,帮助读者更好地掌握这两种流的使用
2023-05-17

C#中如何使用文件IO和流操作进行数据读写

C#中如何使用文件IO和流操作进行数据读写,需要具体代码示例在C#编程中,文件IO和流操作是常用的技术,用于读取和写入文件的数据。无论是处理文本文件、二进制文件,还是读取网络流数据,我们都可以通过文件IO和流操作来实现。文件IO和流操作提供
2023-10-22

C#中如何使用文件IO和流操作进行数据读写及解决方法

C#中如何使用文件IO和流操作进行数据读写及解决方法在开发过程中,我们常常需要进行文件的读写操作。C#提供了丰富的文件IO和流操作,使得数据的读写变得更加灵活和高效。本文将探讨C#中如何使用文件IO和流操作进行数据读写,并提供具体的代码示例
2023-10-22

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录