yo!这里是c++IO流相关介绍

news2024/12/27 3:07:13

目录

前言

C语言的输入输出

C++IO流基本介绍

流的概念

IO流类库

iostream

fstream

stringstream

后记


前言

        学过C语言的输入输出相关知识点的童鞋应该多多少少会觉得有些许麻烦,反正我就是这么觉得的,scanf、printf等函数不仅数量众多,而且转化格式必须匹配,否则会得到错误的结果,有些函数还必须得预留出保存结果得空间,而这个空间又不好界定。那这就是C语言这套面向过程的输入输出相关函数使用的弊端,在c++面向对象的环境中,这些输入输出功能必定会被封装成类,正如大家熟知的cin、cout就是相关类多定义出来的对象,使用起来非常之舒服,那么c++的输入输出就叫做IO流。下面会先回忆一下C语言的输入输出函数,对比之下来拓展学习c++IO流,接下来就开始吧!

C语言的输入输出

        初学C语言时,想必大家一定经常用scanf、printf两个函数输入输出吧,但是对于fprintf、fscanf、sprintf、sscanf等函数,大部分同学应该是见过都很少用过,这些函数是后面学习到字符串和文件两部分知识点才会接触到的函数,也是用于输入输出的。下面我们分别大概介绍一下:

  • scanf:

  • printf:

        其中,第一个参数是一个常量字符指针,第二个参数是可变参数列表,表示用户自定义传入参数个数,返回值int表示按照指定的格式符正确读入/输出的数据的个数,但如果输入/输出数据与指定格式不符,则会产生错误,函数会立即终止,设置错误码,并返回已经成功读取/输入的数据的个数。

  • fscanf:

  • fprintf:

        可见,fprintf、fscanf函数与printf、scanf函数不同的只是多了一个参数——文件指针。回想一下,其实scanf函数默认是从键盘拿数据,因此fscanf函数是从指定文件拿数据,而printf默认是输出到显示屏,fprintf则是输出到指定文件。

  • sscanf:

  • sprintf:

  • snprintf:

        对于sscanf、sprintf函数也是一样,不同的地方在于多了一个参数——常量字符指针,因此sscanf是从一个字符串拿数据,而sprintf是将数据按照指定格式输出到一个字符串当中。snprintf更多了一个参数——字符个数,可以指定输入到字符串中的字符个数。

C++IO流基本介绍

  • 流的概念

        是对一种有序连续且具有方向性的数据( 其单位可以是bit、byte、packet )的抽象描述,而C++流是指信息从外部输入设备(如键盘)向计算机内部(如内存)输入和从内存向外部输出设备(显示器)输出的过程,这种过程被形象的比喻为“流”,具有有序连续、方向性的特性。

  • IO流类库

        为了实现这种流动,C++定义了I/O标准类库,如下图,这些每个类都称为流/流类,用以完成某方面的功能。其中ios为基类,其他类都是直接或间接派生自ios类。其中,不看第三列,派生类分为三种——iostream、fstream、stringstream,而第三列的cin、cout等是iostream定义出的对象,其中cin、cout也是我们自学习c++以来经常使用的输入输出流。宏观来看,iostream是对scanf、printf的替换,fstream是对fscanf、fprintf的替换,stringstream是对sscanf、sprintf的替换,下面分别介绍:

iostream

        iostream类库是istream、ostream类库合并而来,没什么别的意义,就是定义了这一个头文件,两个类的功能都可以使用。istream是输入流类,定义出cin进行标准输入即数据通过键盘输入到程序中,ostream是输出流类,定义出cout、cerr、clog,其中cout进行标准输出,即数据从内存流向显示屏,cerr用来进行标准错误的输出,以及clog进行日志的输出,这三个对象现在基本没有区别,只是应用场景不同。下面强调几个使用过程中的重点注意事项。

        1.空格和回车都可以作为数据之间的分格符,因此多个数据可以在一行输入,也可以分行输 入。但同时数据如果是字符型和字符串,则空格将无法用cin输入,也不能将空格通过键盘输入到字符串中。 

eg(输入多个值的方法):

如果遇到数据之间没有加空格或换行的情况呢?可能c++方法要比C语言方法稍复杂一点。

        2.在笔试一些oj题时,有很多题目需要循环输入得到多个数据,c++的方法就很简单,对于一个值循环输入:

while(cin>>a)
{
    //...
}

对于多个值循环输入:

while(cin>>a>>b>>c)
{
    //...
}

        大部分oj题的输入数据都是用空格或换行分割,因此使用以上方法拿到数据绝对没有任何问题。此外还值得提一嘴的是在vs系列编译器下连续输入时输入ctrl+z停止输入。

3. istream类型转换为逻辑判断值

        为什么上面的cin>>a在输入结束之后会停止循环,那肯定是while条件为假,但是cin>>a之后的返回值依旧是istream类型(istream& operator>> (int& val);),这该如何知道这是真是假呢?其实在istream类型当中,有这样一个成员函数——explicit operator bool() const{},不看explicit和const两个关键字,可以看到这个函数是没有返回值的,其实并不是,这是它的特殊实现方式,返回值就是bool类型,因为我们需要一个逻辑判断值,所以在operator后加上bool,然后函数体内就可以实现将istream类型转bool值得逻辑,实则就是看指向键盘数据的标识符是否指向结尾,指向结尾说明拿到了所有的数据,之后返回false,while循环条件拿到false则循环输入结束。这样说有点混乱,因为拿不到istream类型的实现机制,下面举两个例子:

1)如下图,类A我们实现了构造函数,那么我们相当于实现了将一个内置类型转换为一个自定义类型的功能,其实就是一种隐式类型转换;此时我们也想实现将一个自定义类型转换为一个内置类型的功能,比如说转换为int类型,那么我们就可以operator加上int实现这样一个函数,函数体内写上转为内置类型的逻辑,也就是拿到成员变量,将其返回出来,实现成功。

eg1:

 2)以下代码块实现一个Date类型和能够将其输入输出的operator重载函数。其中我们想要实现一个将这个Date类型转换为bool类型的功能,就是成员函数operator加上bool,函数体内实现转换逻辑,这里比如说是看year变量,为0返回false,为非0返回true,如此就实现了将自定义类型转内置类型的功能。

eg2:

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)
	{}
	operator bool()
	{
		// 这里是随意写的,假设输入_year为0,则结束
		if (_year == 0)
			return false;
		else
			return true;
	}
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;
}

fstream

        不同于iostream的是,c++并没有提供像cin、cout这样的全局对象供我们使用,这里就需要我们自己去定义,包含头文件<fstream>后,三种定义方式如下:

  • 只输入用:ifstream 对象名(文件名)
  • 只输出用:ofstream 对象名(文件名)
  • 输入输出两个都能用:fstream 对象名(文件名)

        定义好之后,对象的用法就如同cin、cout一样了。举个例子,定义一个类ServerInfo作为文件IO的数据,包括ip地址、端口号、时间(前面模拟实现Date类),再定义一个类Consoler对一个文件实现文本读、文本写、二进制读、二进制写的功能,构造函数就是传入一个文件名即可。

        因为文件分为文本文件和二进制文件,所以有四个读写函数。对于二进制写,ofstream定义一个对象ofs,其中第一个参数就是文件名,第二个参数表示是二进制读写,之后使用write函数将信息数据写入文件;二进制读也是一样,使用read函数将信息数据读出来;对于文本写,不需要加第二个参数,因为默认就是文本读写,之后使用如同cin、cout的形式读写数据。注意要分清:ifs对应cin,cin是将键盘数据读到内存中,而ifs是将文件数据读到内存中;ofs对应cout,cout是将内存数据写到显示屏,而ifs是将内存数据写到文件中。

        综上,二进制文件和文本文件的读写方式不一样,c++文件流的优势就是可以对内置类型和自定义类型,都使用一样的方式,去流插入和流提取文件数据,不过前提是要自定义类型需要重载>> 和 <<。

eg:

class ServerInfo
{
public:
	char _ipaddr[32];
	int _port;
	Date _date;
};

class Consoler
{
public:
	Consoler(const char* filename)
		:_filename(filename)
	{}

	//二进制写
	void WriteBin(const ServerInfo& info)
	{
		ofstream ofs(_filename, ios_base::out | ios_base::binary);
		/*ofs << info._ipaddr << endl;
		ofs << info._port << endl;
		ofs << info._date << endl;*/   //这样写的话还是文本写

		ofs.write((char*)&info, sizeof(info));
	}

	//二进制读
	void ReadBin(ServerInfo& info)
	{
		ifstream ifs(_filename, ios_base::out | ios_base::binary);
		ifs.read((char*)&info, sizeof(info));

		cout << info._ipaddr << endl;
		cout << info._port << endl;
		cout << info._date << endl;
	}

	//文本写
	void WriteText(const ServerInfo& info)
	{
		ofstream ofs(_filename);
		ofs << info._ipaddr << endl;
		ofs << info._port << endl;
		ofs << info._date << endl;
	}

	//文本读
	void ReadText(ServerInfo& info)
	{
		ifstream ifs(_filename);
		ifs >> info._ipaddr >> info._port >> info._date;

		cout << info._ipaddr << endl;
		cout << info._port << endl;
		cout << info._date << endl;
	}
private:
	string _filename;
};


int main()
{
	ServerInfo info = { "192.168.1.1",28,{2024,1,26} };

	Consoler clr("draft.txt");

	clr.ReadBin(info); //二进制读
	clr.WriteBin(info); //二进制写
	
	clr.ReadText(info); //文本读
	clr.WriteText(info); //文本写

	return 0;
}

stringstream

        stringstream库的学习可以参考fstream,只不过第一个参数不是文件指针,而是string类型的对象,包含<sstream>后,就可以将数据输入到一个string中,也可以将string中的数据读出来。stringstream库分为istringstream库和ostringstream库,使用stringstream定义的对象可以读也可以写,使用istringstream定义的对象只可以写,使用ostringstream定义的对象只可以读,下面举几个例子简单说明一下用法。

1.不同数值类型与string之间的转换

eg:

#include <iostream>
#include <sstream>
#include <string>

using namespace std;

int main()
{
	//stringstream oss;   //一开始字符串流中啥也没有
	//stringstream oss("8888");   //一开始字符串流中就存在一个8888的字符串,默认是覆盖写
	stringstream oss("8888 ", ios_base::out | ios_base::ate);   //追加写
	int i = 123;
	double d = 3.14;
	string s = "hello";

	oss << i <<" " << d <<" " << s;

	cout << oss.str() << endl;
	return 0;
}

2.序列化和反序列化

        在通过网络传递信息的过程中,我们是需要一个string表示各种信息,比如说qq中发送的一条信息,包括发送者、发送对象、发送时间、发送内容等等,将这些信息序列化成一个字符串发送过来,接收后将这个字符串反序列化出来得到这些信息,这个过程就可以用到stringstream类。 

eg:

class QQInfo
{
public:
	string _name;
	Date _date;
	string _msg;
};

int main()
{
	QQInfo info = { "张三",{2024,2,2},"你吃饭了没?" };

	//序列化
	stringstream oss;
	oss << info._name << " " << info._date << " " << info._msg;
	string str = oss.str();

	//通过网络发送这个str
	//...
	
	//反序列化
	QQInfo info2;
	stringstream iss(str);
	iss >> info2._name >> info2._date >> info2._msg;
	
	//验证
	cout << info2._name << " | " << info2._date << " | " << info2._msg << endl;

	return 0;
}

注意:

        1.stringstream实际上是在其底层维护了一个string类型的对象

        2.可以使用s. str("")方法将底层string对象设置为空字符

        3.可以使用s.str()将让stringstream返回其底层的string对象

        总的来说,stringstream使用string类对象代替字符数组,可以避免缓冲区溢出的危险,而且其会对参数类型进行推演,不需要格式化控制,也不会出现格式化失败的风险,因此使用更方便,更安全

后记

        从上面的学习可以看出,c++IO流是一个庞大的类库,很多类都是直接或间接派生基类所得,非常符合c++的世界观,使用起来非常方便,基类能支持的,子类都可以使用。此外,无论是输入输出方式、格式、类型安全、缓冲区和错误处理,C++的IO流相较于C语言的输入输出更加高级、安全和方便使用。而且从学习成本上看,当我们学会使用iostream类的使用,fstream、string stream类的使用也是得心应手。因此建议多多使用c++IO流来输入输出,拜拜!


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1430419.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

RNN实战具体跑的代码

一、首先先上代码&#xff1a;这个是接pytorch API的调用代码 import torch import torch.nn as nn bs, T 2,3#批次大小&#xff0c;输入序列长度 input_size,hidden_size2,3#输入特征大小&#xff0c;隐含层特征大小 input torch.randn(bs,T, input_size) h_prev torch.ze…

css新手教程

css新手教程 课程&#xff1a;14、盒子模型及边框使用_哔哩哔哩_bilibili 一.什么是CSS 1.什么是CSS Cascading Style Sheet 层叠样式表。 CSS&#xff1a;表现&#xff08;美化网页&#xff09; 字体&#xff0c;颜色&#xff0c;边距&#xff0c;高度&#xff0c;宽度&am…

用户体验优化:HubSpot的秘密武器

在当今数字化市场中&#xff0c;提升用户体验已经成为企业成功的关键因素之一。HubSpot&#xff0c;作为一款领先的营销自动化工具&#xff0c;不仅在推动销售业绩上表现出色&#xff0c;同时通过其独特的策略也致力于提升用户体验。运营坛将深入探讨HubSpot是如何通过个性化推…

Leetcode92:反转链表II(区间反转链表)

一、题目 给你单链表的头指针 head 和两个整数 left 和 right &#xff0c;其中 left < right 。请你反转从位置 left 到位置 right 的链表节点&#xff0c;返回 反转后的链表 。 示例&#xff1a; 输入&#xff1a;head [1,2,3,4,5], left 2, right 4 输出&#xff1a…

Redisson看门狗机制

一、背景 网上redis分布式锁的工具方法&#xff0c;大都满足互斥、防止死锁的特性&#xff0c;有些工具方法会满足可重入特性。如果只满足上述3种特性会有哪些隐患呢&#xff1f;redis分布式锁无法自动续期&#xff0c;比如&#xff0c;一个锁设置了1分钟超时释放&#xff0c;…

Servlet+Ajax实现对数据的列表展示(极简入门)

目录 1.准备工作 1.数据库源&#xff08;这里以Mysql为例&#xff09; 2.映射实体类 3.模拟三层架构&#xff08;Dao、Service、Controller&#xff09; Dao接口 Dao实现 Service实现&#xff08;这里省略Service接口&#xff09; Controller层&#xff08;或叫Servlet层…

Vulnhub billu b0x

0x01 环境搭建 1. 从官方下载靶机环境&#xff0c;解压到本地&#xff0c;双击OVF文件直接导入到vmware虚拟机里面。2. 将虚拟机的网络适配器调成NAT模式&#xff0c;然后开机即可进行操作了。 0x02 主机发现 nmap -sn 192.168.2.0/24 成功获取靶机IP为192.168.2.129。 0x0…

网络时间协议NTP工作模式

单播服务器/客户端模式 单播服务器/客户端模式运行在同步子网中层数较高层上。这种模式下,需要预先知道服务器的IP地址。 客户端:运行在客户端模式的主机(简称客户端)定期向服务器端发送报文,报文中的Mode字段设置为3(客户端模式)。当客户端接收到应答报文时,客户端会…

指针2 1月31日学习笔记

一、strncpy、strncmp、strncat函数 strncpy函数用于将一个字符串的一部分拷贝到另一个字符串中。 char* strncpy(char *dest, const char *src, size_t n){size_t i;for (i 0; i < n && src[i] ! \0; i)dest[i] src[i];for ( ; i < n; i)dest[i] \0;return …

2024牛客寒假算法基础集训营1部分题解

// 能力有限&#xff0c;做多少发多少。 A-DFS搜索 题目描述 最近&#xff0c;fried-chicken完全学明白了DFS搜索&#xff08;如上图所示&#xff09;&#xff01;于是学弟向他请教DFS搜索&#xff0c;fried-chicken热心的进行了讲解&#xff1a; 所谓DFS搜索&#xff0c;就…

上海纽约大学信息技术部高级主任常潘:解密大数据引领的未来教育革命

大数据产业创新服务媒体 ——聚焦数据 改变商业 在数字化时代&#xff0c;大数据技术的应用已经深刻地改变着各行各业。特别是在教育领域&#xff0c;智慧校园建设作为现代化校园的代名词&#xff0c;正迎来大数据技术的巨大机遇。 1月17日&#xff0c;上海纽约大学信息技术部…

嵌入式软件工程师面试题——嵌入式专题(五十二)

说明&#xff1a; 面试群&#xff0c;群号&#xff1a; 228447240面试题来源于网络书籍&#xff0c;公司题目以及博主原创或修改&#xff08;题目大部分来源于各种公司&#xff09;&#xff1b;文中很多题目&#xff0c;或许大家直接编译器写完&#xff0c;1分钟就出结果了。但…

EtherCAT转ModbusTCP网关

一、功能概述 1.1设备简介 本产品是EtherCAT和Modbus TCP网关&#xff0c;使用数据映射方式工作。 本产品在EtherCAT侧作为EtherCAT从站&#xff0c;接TwinCAT、CodeSYS、PLC等&#xff1b;在ModbusTCP侧做为ModbusTCP主站&#xff08;Client&#xff09;或从站&#xff08;…

【蓝桥杯冲冲冲】[NOIP2003 普及组] 数字游戏

蓝桥杯备赛 | 洛谷做题打卡day25 文章目录 蓝桥杯备赛 | 洛谷做题打卡day25[NOIP2003 普及组] 数字游戏题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 提示思路 题解代码我的一些话 [NOIP2003 普及组] 数字游戏 题目描述 丁丁最近沉迷于一个数字游戏之中。这个游戏看…

MySQL for update锁表还是锁行校验

select * from user where id 1 for update ; 1. for update作用 在MySQL中&#xff0c;使用for update子句可以对查询结果集进行行级锁定&#xff0c;以便在事务中对这些行进行更新或者防止其他事务对这些行进行修改。 当使用for update时&#xff0c;锁定行的方式取决于wh…

【初中生讲机器学习】4. 支持向量机算法怎么用?一个实例带你看懂!

创建时间&#xff1a;2024-02-02 最后编辑时间&#xff1a;2024-02-03 作者&#xff1a;Geeker_LStar 你好呀~这里是 Geeker_LStar 的人工智能学习专栏&#xff0c;很高兴遇见你~ 我是 Geeker_LStar&#xff0c;一名初三学生&#xff0c;热爱计算机和数学&#xff0c;我们一起加…

【八大排序】冒泡排序 | 快速排序 + 图文详解!!

&#x1f4f7; 江池俊&#xff1a; 个人主页 &#x1f525;个人专栏&#xff1a; ✅数据结构冒险记 ✅C语言进阶之路 &#x1f305; 有航道的人&#xff0c;再渺小也不会迷途。 文章目录 交换排序一、冒泡排序1.1 算法步骤 动图演示1.2 冒泡排序的效率分析1.3 代码实现1.4 …

HSRP配置指南

实验大纲 第 1 部分&#xff1a;验证连通性 步骤 1&#xff1a;追踪从 PC-A 到 Web 服务器的路径 步骤 2&#xff1a;追踪从 PC-B 到 Web 服务器的路径 步骤 3&#xff1a;观察当 R3 不可用时&#xff0c;网络的行为 第 2 部分&#xff1a;配置 HSRP 主用和 备用路由器 步…

【Crypto | CTF】BUUCTF rsarsa1

天命&#xff1a;第二题RSA解密啦&#xff0c;这题比较正宗 先来看看RSA算法 这道题给出了 p&#xff0c;q&#xff0c;E&#xff0c;就是给了两个质数和公钥 有这三个东西&#xff0c;那就可以得出私钥了 最后把私钥和质数放进去解密即可得到解密后的明文 from gmpy2 impor…

公交最短距离-算法

题目 给定一个一维数组&#xff0c;其中每一个元素表示相邻公交站之间的距离&#xff0c;比如有四个公交站A,B,C,D&#xff0c;对应的距离数组为&#xff0c;1,2,3,4&#xff0c;如下图示 给定目标站X和Y&#xff0c;求他们之间最短的距离 解题 遍历一次整个数组&#xff0c;…