C++输入输出与IO流

news2024/11/24 18:53:47

C++ 输入输出与I/O流

文章目录

  • C++ 输入输出与I/O流
    • IO类型与基础特性
      • 概念与特性
      • IO状态
      • 输出缓冲区
    • 文件输入输出
      • 文件模式
    • string流
    • IO处理中常用的函数及操作符
    • 综合练习与demo
      • 一、 创建文件并写入
      • 二、控制台输入数据并拆分存储
      • 三、读写电话簿

IO类型与基础特性

C++11标准提供了几种IO处理操作。其中最为熟悉的就是控制台IO:iostream,除此之外,还提供了文件操作IO:fstream以及string处理IO:stringstream,这两种IO操作都继承自iostream,因此在iostream上可以执行的操作,在另外两种IO类型中亦可执行。(i继承i,o继承o)

IO库类型及头文件

头文件类型作用
iostreamistream, wistream从流读取数据
ostream, wostream向流写入数据
iostream, wiostream读写流
fstreamifstream, wifstream从文件中读取数据
ofstream, wofstream向文件写入数据
fstream, wfstream读写文件
sstreamistringstream, wistringstream从string中读取数据
ostringstream, wostringstream向string中写入数据
stringstream, wstringstream读写string

注:w开头的类型用于操纵wchar_t类型的数据。

  • cin:一个istream对象,从标准输入中读取数据
  • cout: 一个ostream对象,向标准输出写入数据
  • cerr:一个ostream对象,通常用于输出程序错误信息,写入到标准错误
  • >>:用来从一个istream对象读取输入数据
  • <<: 用来向一个ostream对象写入输出数据
  • eof(): 用来判断流是否到达末尾

概念与特性

  • IO对象不能被拷贝或赋值
  • 如果程序崩溃,输出缓冲区是不会被刷新的。
  • 当一个输入流被关联到一个输出流时,任何试图从输入流读取数据的操作都会先刷新关联的输出流

IO状态

strm::iostate是C++中表示流的状态的一种类型,是一个枚举类型。(这里的strm是一种IO类型,如ostreamfstream等)

流状态类型说明
strm::badbit**表示流发生了无法恢复的错误,通常是系统级错误。**如果其对应置位被clear()清除,也只是将流状态恢复为有效,不会解决问题。
strm::falibit表示流操作失败,这种错误通常是可恢复的
strm::eofbit表示流到达了文件结束
strm::goodbit表示流处于正常状态,所有的标志位都是0

标准库提供了一组函数用来查询这些标志位的状态。其中good()fail()是确定流总体状态的方法,当我们将流当做条件使用,如while(cin >> word)时,其代码就等价于while(!cin.fail())

状态查询方法说明
s.eof()若流s的eofbit置位,则返回true
s.fail()若流s的falibitbadbit置位,则返回true
s.bad()若流s的badbit置位,则返回true
s.good()若流s处于有效状态,则返回true
s.clear()将流s中所有条件状态复位,将流的状态设置为有效。void类型
s.clear(flags)复原指定的flags标志位。void类型
s.setstate(flags)复原给定的flags标志位。void类型
s.rdstate()返回的是当前流的状态,以 iostate 枚举值表示。

输出缓冲区

每个输入流都管理一个缓冲区,用来保存程序读写的数据。

当程序正常结束、缓冲区满、手动指定缓冲(如操作符endl,cerr等)缓冲区都会被刷新。

操作符说明
std::endl换行并刷新缓冲区
std::flush刷新缓冲区,且不附加额外字符
std::ends输出空字符,并刷新缓冲区

unitbuf与nounitbuf

cout << unitbuf; 		// 所有输出操作后都会立刻刷新缓冲区
// 任何输出都立刻刷新,无缓冲
...
cout << nounitbuf;      // 恢复常规模式

文件输入输出

fstream特有的操作

操作说明
fstream fstrm;创建一个未绑定的文件流
fstream fstrm(s);创建一个文件流,并打开名为s的文件(s可为string,亦可为C风格字符串之指针)。此构造函数是explicit的
fstream fstrm(s, mode)与上者类似,按指定mode打开指定s文件
fstrm.open(s)打开名为s的文件,并将文件与fstrm绑定
fstrm.close()关闭与fstrm绑定的文件。void类型
fstrm.is_open()返回一个bool值,说明关联文件是否成功打开且尚未关闭
  • 当一个fstream对象被销毁时,close会自动被调用

文件模式

文件模式,用于指出文件流如何使用文件

文件模式说明
in以读的方式打开
out以写的方式打开
app每次写操作之前均定位到文件末尾
ate打开文件后立即定位到文件末尾
trunc截断文件
binary以二进制方式进行IO
  • ifstream默认in模式,ofstream默认out模式,fstream默认inout模式
  • out只支持ofs和fs流; in只支持ifs和fs流
  • out被设定时,才可以设定trunc
  • trunc没被设定时,才可以设定app模式
  • app模式下,没有显式指定out,文件依然以输出方式被打开
  • atebinary模式可用于任何类型的文件流对象

注意1:以out模式打开文件会丢弃已有数据

  • 使用ofstream打开一个文件,会将文件原有的数据丢弃。可以同时采用显式指定appin的方式打开,避免造成数据丢失。

注意2:每次调用open是都会确认文件模式

  • 我们使用同一个ofstream流打开文件,在第一个文件指定了什么模式打开,并不会在第二个文件中依然存在,只会选择默认方式
  • 所以我们利用与out相关的文件流打开文件时,一定要注意最好显式指定文件模式

我们使用文件流时通常包含以下步骤

  • 创建一个文件流
    • 默认文件流
    • 指定绑定文件(打开or不打开)
  • 打开文件
    • 显式指定打开模式
  • 关闭文件
void fstreamTest(){
    string filename = "F:xxxxxx\\test.txt";
    // 创建一个文件流对象
    std::fstream fstrm(filename);
    // fstrm.open(filename);  也可以创建一个默认的fstream实例,用这种方式打开
    
    string line;
    // 是否成功打开与其关联的文件
    if(fstrm.is_open()){
        while(std::getline(fstrm, line)){
            std::cout<< line << std::endl;
        }
    }else{
        std::cerr<<"Flie is not open!"<<std::endl;
    }
    // 最后用完关闭
    fstrm.close();
}

string流

stringstream特有操作

操作说明
sstream strm一个未绑定的string流对象
sstream strm(s)创建一个string流对象,并保存string s的拷贝(该构造函数explicit)
strm.str()返回strm保存的string的拷贝
strm.str(s)将s拷贝到strm中。void类型

IO处理中常用的函数及操作符

getline(cin, str)

  • getline函数会从输入流中读取字符,直到遇到换行符或者指定分隔符位置,会将读取到的字符存储在str中
template <class charT, class traits, class Allocator>
std::basic_istream<charT,traits>& getline (std::basic_istream<charT,traits>& is, std::basic_string<charT,traits,Allocator>& str, charT delim);
  • is 是输入流,例如 std::cin
  • str 是要存储读取到的一行数据的字符串。
  • delim 是可选参数,用于指定行结束的分隔符,默认为换行符 \n

std::boolalpha

  • 使控制台在输入bool类型数据时,输出字符型的truefalse,而非0和1
  • 关闭使用:std::noboolalpha
void boolalphaTest(){
    std::cout<<"Default bool type: "<<true<<" "<<false<<'\n'
            <<std::boolalpha
            <<"use boolalpha: "<< true<<" "<<false<<'\n'
            <<std::noboolalpha
            <<"close boolalpha: "<< true<<" "<<false<<'\n';
}

image-20240429223253041

综合练习与demo

一、 创建文件并写入

	std::ofstream outFile;			// 创建写入文件流
	outFile.open("emptable1.txt"); 	// 创建该txt文件
	if (!outFile)
	{
		std::cerr << "无法打开文件" << std::endl;
		return;
	}
	outFile << "on, ";				// 写入数据
	for (auto s : stepMap)
	{
		outFile << s.second << ", ";
	}
	outFile << endl;
	outFile.close();				// 关闭文件

二、控制台输入数据并拆分存储

/*1、输入一行数据并按空格拆分*/
string input;
vector<string> latex;
cout << "Please input: ";
// 将一行输入存储到input中
std::getline(std::cin, input);		
// 创建istring流,并将input拷贝过去
std::istringstream iss(input);				
string str;
// istring流将通过空格进行拆分,传递赋值给str
while (iss >> str)							
{
    latex.push_back(str);
}

/*2、输入多行数据并按行存储*/
 std::vector<std::string> lines;
std::string line;

std::cout << "Enter multiple lines of text (enter an empty line to finish):" << std::endl;

// 使用 getline 在循环中读取每一行输入
while (true) {
    std::getline(std::cin, line);
    if (line.empty()) {
        // 当输入为空行时,结束输入
        break;
    }
    lines.push_back(line);
}

三、读写电话簿

给定一个电话簿文件,里面格式大致为name number1 number2

读取该文件,并且存储人物及电话号码信息,存储时判断号码是否符合规定

将正确格式化后的字符,输出到结果文件中(结果文件包含title,不能覆盖)

image-20240429230735986

原始读取文件

image-20240429230808682

原始存储文件
#include <fstream>
#include <iostream>
#include <vector>
#include <sstream>
using namespace std;

struct PersonInfo
{
    /* data */
    string name;
    vector<string> phones;
};

// 检查电话号码是否合法
bool checkNumber(string s)
{
    if (s.size() != 11 && (s.size() < 4 || s.substr(0, 4) != "0831"))
    {
        return false;
    }
    else
    {
        for (auto &c : s)
        {
            if (!isdigit(c) && c != '-')
            {
                return false;
            }
        }
    }
    return true;
}

// 阅读最初的电话簿并存储
vector<PersonInfo> getOriginNumbers(string s)
{
    ifstream ifstm;
    // 文件输入流以默认模式打开文件
    ifstm.open(s);
    vector<PersonInfo> people;
    if (ifstm.is_open())
    {
        string line, number;
        // line以行存储文件中的内容
        while (getline(ifstm, line))
        {
            PersonInfo info;
            // 使用istringstream流时,record会根据空白符号对string进行拆分
            istringstream record(line);
            // 录入名字
            record >> info.name;
            // 录入多个号码
            while (record >> number)
            {
                info.phones.push_back(number);
            }
            people.push_back(info);
        }
    }
    else
    {
        cerr << "Open file is faild! " << endl;
    }
    ifstm.close();
    return people;
}

// 将正确的电话存储到电话簿中
void saveRightNumbers(vector<PersonInfo> &people, string s)
{
    ofstream ofs;
    // 以追加形式打开,避免覆盖文件中原有的内容
    ofs.open(s, ofstream::app);
    
    if (ofs.is_open())
    {
        for (const PersonInfo &p : people)
        {
            ofs<<'\n';      // 先换行
            // 记录哪些电话号码有误
            ostringstream badNums;
            ofs << p.name<<"    ";
            for (const auto &number : p.phones)
            {
                // 如果电话号码错误,存储到该string输出流中
                if (!checkNumber(number))
                {
                    // 存储多个错误号码
                    badNums << " " << number;
                }else{
                    ofs<<number << "    ";
                }
            }
            if (!badNums.str().empty())
            {
                // 输出谁的电话号码有问题
                cerr << "[input error]  name: " << p.name
                     << ",  invalid number(s): " << badNums.str() << endl;
            }
        }
    }
    else
    {
        cerr << "Open savefile is faild! " << endl;
    }
    ofs.close();
}

int main()
{
    string originfile = "F:\\xxxxxx\\numberDict.txt";
    string savefile = "F:\\xxxxxxx\\save.txt";
    vector<PersonInfo> people = getOriginNumbers(originfile);
    saveRightNumbers(people, savefile);
    return 0;
}

image-20240429231037247

控制台输出

image-20240429231057014

运行后的save文件

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

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

相关文章

【ArcGIS微课1000例】0117:ArcGIS中如何将kml(kmz)文件转json(geojson)?

文章目录 一、kml获取方式二、kml转图层三、图层转json一、kml获取方式 kml文件是一种很常用的数据格式,可以从谷歌地球(googleearth)获取某一个地区的kml范围文件,如青海湖(做好的kml文件可以从配套实验数据包0117.rar中获取)。 二、kml转图层 打开【KML转图层】工具,…

前端项目打包、部署的基础 (vue)

详细请看B站视频 BV19n4y1d7Gr 《禹神&#xff1a;前端项目部署指南&#xff0c;前端项目打包上线》&#xff0c;本博客为自用视频笔记。 目录 项目打包vue打包打包前分析项目请求 本地服务器部署问题 & 解决问题1&#xff1a;刷新页面404问题问题2&#xff1a;ajax请求废…

最新下载:PDFFactoryFinePrint【软件附加安装教程】

简介&#xff1a; pdfFactory是一款无须 Acrobat 创建 Adobe pdf 文件的打印机驱动程序&#xff0c; 提供的创建 PDF 文件的方法比其他方法更方便和高效。 pdfFactory 支持从所有应用程序轻松、可靠地创建 PDF 文件。 支持将单页或两页的文档&#xff0c;直接打印为PDF文件&a…

信息学奥赛初赛天天练-24-二叉树、N叉树遍历技巧与前缀表达式、中缀表达式、后缀表达式应用实战演练

PDF文档公众号回复关键字:20240609 单项选择题(共15题,每题2分,共计30分:每题有且仅有一个正确选项) 5 根节点的高度为1,一根拥有2023个节点的三叉树高度至少为( )。 A 6 B 7 C 8 D 9 8 后缀表达式 6 2 3 + - 3 8 2 / + * 2 ^ 3 + 对应的中缀表达式是( ) A ((…

[Linux]内网穿透nps

文章目录 基础文件下载项目地址下载地址 客户端安装解压文件客户端启动客户端注册到linux系统服务客户端注册到windows系统服务windows bat 一键管理员注册windows bat 一键管理员取消 基础文件下载 项目地址 https://github.com/ehang-io/nps 下载地址 Releases ehang-io…

12.打渔还是晒网

上海市计算机学会竞赛平台 | YACSYACS 是由上海市计算机学会于2019年发起的活动,旨在激发青少年对学习人工智能与算法设计的热情与兴趣,提升青少年科学素养,引导青少年投身创新发现和科研实践活动。https://www.iai.sh.cn/problem/17 题目描述 有句俗话叫“三天打渔,两天…

VBA高级应用30例应用2实现在列表框内及列表框间实现数据拖动

《VBA高级应用30例》&#xff08;版权10178985&#xff09;&#xff0c;是我推出的第十套教程&#xff0c;教程是专门针对高级学员在学习VBA过程中提高路途上的案例展开&#xff0c;这套教程案例与理论结合&#xff0c;紧贴“实战”&#xff0c;并做“战术总结”&#xff0c;以…

AI大模型学习(非常详细)零基础入门到精通,收藏这一篇就够了

前言 随着人工智能技术的快速发展&#xff0c;AI大模型学习正成为一项备受关注的研究领域。为了提高模型的准确性和效率&#xff0c;研究者们需要具备深厚的数学基础和编程能力&#xff0c;并对特定领域的业务场景有深入的了解。通过不断优化模型结构和算法&#xff0c;AI大模…

【西瓜书】9.聚类

聚类任务是无监督学习的一种用于分类等其他任务的前驱过程&#xff0c;作为数据清洗&#xff0c;基于聚类结果训练分类模型 1.聚类性能度量&#xff08;有效性指标&#xff09; 分类任务的性能度量有错误率、精度、准确率P、召回率R、F1度量(P-R的调和平均)、TPR、FPR、AUC回归…

关系数据库标准查询语言-SQL-SQL语言概述

一、SQL(Structured Query Language)语言 1、是高度非过程化的语言 2、关系数据库管理系统(RDBMS)都支持SQL标准 3、具有定义、查询、更新、控制四大功能 4、数据库对象由数据库&#xff08;Database&#xff09;、基本表&#xff08;Table&#xff09;、视图&#xff08;V…

Spark参数配置不合理的情况

1.1 内存设置 &#x1f4be; 常见的内存设置有两类&#xff1a;堆内和堆外 &#x1f4a1; 我们作业中大量的设置 driver 和 executor 的堆外内存为 4g&#xff0c;造成资源浪费 &#x1f4c9;。 通常 executor 堆外内存在 executor.cores1 的时候&#xff0c;1g 足够了&…

都怪我当初没有好好了解你,Java虚拟机(JVM)

初始JVM JVM本质是一个运行在计算机上的程序&#xff0c;作用是运行Java字节码文件。 下面是它的运行流程&#xff1a; 看完上述运行过程&#xff0c;现在提出一个问题&#xff1a;Java是编译型语言还是解释型语言&#xff1f; 这里先补充什么是编译&#xff0c;什么是解释&am…

电子阅览室有何作用

随着互联网的快速发展&#xff0c;电子阅览室逐渐成为人们获取知识的新方式。它为读者提供了便捷、高效的阅读体验&#xff0c;具有诸多作用。首先&#xff0c;电子阅览室拥有丰富的电子书籍资源&#xff0c;涵盖了各个领域的知识。无论是文学作品还是学术论文&#xff0c;读者…

(Proteus仿真设计)基于51单片机的电梯程序控制系统

&#xff08;Proteus仿真设计&#xff09;基于51单片机的电梯程序控制系统 一.项目介绍 本设计模拟的是一个五层的&#xff0c;各楼层间隔为4.5m的电梯程序控制系统&#xff0c;能够完成各楼层乘客的接送任务。形象地说&#xff0c;就是要对不同楼层乘客的不同需求&#xff0…

Python第二语言(八、Python包)

目录 1. 什么是Python包 2. 创包步骤 2.1 new包 2.2 查看创建的包 2.3 拖动文件到包下 3. 导入包 4. 安装第三方包 4.1 什么是第三方包 4.2 安装第三方包-pip 4.3 pip网络优化 1. 什么是Python包 包下有__init__.py就是包&#xff0c;无__init__.py就是文件夹。于Ja…

Python中报错提示:TypeError: Student() takes no arguments

Python中报错提示&#xff1a;TypeError: Student() takes no arguments 在Python编程中&#xff0c;类是创建对象的蓝图。每个类都可能包含一个特殊的方法__init__&#xff0c;我们称之为构造函数&#xff0c;它在创建新实例时被调用。如果你在尝试创建一个类的实例时遇到了Ty…

SpringBoot+Vue墙绘产品展示交易平台(前后端分离)

技术栈 JavaSpringBootMavenMySQLMyBatisVueShiroElement-UI 角色对应功能 用户管理员 功能截图

想在VBA软件中做个登录验证会员授权,用什么云服务器好?

想在VBA中做个登录验证会员授权&#xff0c;用什么服务器好&#xff1f; 腾讯云99起&#xff0c;百度云50元起&#xff0c;不过也不知道到底是一整个虚拟机服务器&#xff0c; 装了WIN2012系统的&#xff0c;还是只是一个虚拟网站只给你一个文件夹可以上传PHP,ASP网页后台。 价…

知识图谱的应用---智慧交通

文章目录 智慧交通典型应用 智慧交通 现代城市发展过程中的一大问题是交通拥堵&#xff0c;为解决城市发展中的这一顽疾&#xff0c;有必要以现代化高科技技术为支撑&#xff0c;建造城市中的智慧交通系统&#xff0c;从源头入手缓解城市拥挤问题。当前&#xff0c;“智慧交通”…

6月7号作业

1&#xff0c; 搭建一个货币的场景&#xff0c;创建一个名为 RMB 的类&#xff0c;该类具有整型私有成员变量 yuan&#xff08;元&#xff09;、jiao&#xff08;角&#xff09;和 fen&#xff08;分&#xff09;&#xff0c;并且具有以下功能&#xff1a; (1)重载算术运算符…