C++STL开发温习与总结(六): 6.C++语言输入/输出流定义之输入/输出格式控制

简介: C++STL开发温习与总结(六): 6.C++语言输入/输出流定义之输入/输出格式控制

原博主博客地址:http://blog.csdn.net/qq21497936

本文章博客地址:http://mp.blog.csdn.net/postedit/79177645


C++STL开发温习与总结(六):

6.C++语言输入 /输出流定义之输入 /输出格式控制


       本章开始正式进入标准库的温习。


1.C++语言输入/输出流概述


       在C++语言系统中所有的输入/输出流操作都是借助ios类及其派生类对象实现的。ios派生类ostream和istream。cout是ostream的一个对象,cin是istream的一个对象。此两对象的特殊之处在于它们是编译器直接认识的少数几个系统级的对象,其包含于头文件<iosteam>。

        符号“<<”和“>>”是在类(派生类)中定义的重载运算符。


2.输入/输出格式控制


       输出输出格式控制决定了输入或输出的格式,对于实际系统非常重要,下面列举了输入输出格式控制的几种方式:


2-1枚举常量(ios类中的)


       在根基类ios中定义有3个用户需要使用的枚举类型,由于它们是在公用成员部分定义的,所以其中的每个枚举型常量在加上ios::前缀后都可以为本类成员函数和所有外部函数访问。在3个枚举类型中有一个无名枚举类型。

       第一个定义为无名枚举类型,每个枚举常量都是用于设置控制输入输出格式的标志使用的:

enum {
  skipws,
  left,
  right,
  internal,
  dec,
  oct,
  hex
  showbase,
  showpoint,
  uppercase,
  showpos,
  scientific,
  fixed,
  unitbuf,
  stdio
};
  • skipws:设置标志后,从流中输入数据时跳过当前位置及后面的所有连续的空白字符,从第一个非空白字符起读数,否则不跳过空白字符。(空白字符:空格、\t、\r、\n)。
  • left、right、internal:在指定的域宽内分别对齐左、右、中输出,注意internal使数值的符号按左对齐,数值本身按右对齐。域宽内剩余的字符位置用填充符填充。
  • dec、oct、hex:设置标志后,使以后的数值按照十进制、八进制、十六禁止输出。
  • showbase:设置标志后,使数值输出的前面加上“基指示符”。十进制无,八进制为0,十六禁止为0x。
  • showpoint:强制输出的浮点数中带有小数点和小数尾部的无效数字0.
  • uppercase:使输出的十六进制数和浮点数中使用的字母为大写,缺省为不设置(即小写)。
  • showpos:使输出的整数前带有正好“+”,缺省为不设置,即输出的正数前不带任何符号。
  • scientific、fixed:设置scientific后,浮点数按科学表示法输出;fix设置后,使浮点数按定点表示法输出,只能任设其一。缺省由系统适配自动选择合适的输出表示。
  • unitbuf、stdio:这两个很少使用,未介绍。

       第二个枚举类型open_mode,每个常量规定一种文件打开的方式,在定义文件流和打开文件时使用:

enum open_mode {
  in,
  out,
  ate,
  app,
  trunc,
  nocreate,
  noreplace,
  binary
};

      第三个枚举类型,每个枚举常量用于对文件指针的定位操作上:

enum seek_dir{
  beg,
  cur,
  end
}


2-2成员函数(ios类中的)


       ios类提供成员函数对流的状态进行检测和进行输入输出格式控制等操作,所有成员函数如下:

  • int bad():操作出错时返回非0值。
  • int eof():读取到流中最后的文件结束符时返回非0值。
  • int fail():操作失败返回非0值。
  • void clear():清除bad、eof和fail所对应的标志状态,使之恢复为正常状态值0,使good标志状态恢复为1。
  • char fill():返回当前使用的填充字符。
  • char fill(char c):重新设置流中用于输出数据的填充字符为c的值,返回此前的填充字符。系统预设置填充字符为空格
  • long flags():返回当前用于I/O控制的格式状态字。
  • long flags(long f):重新设置格式状态字为f的值,返回此前的格式状态字。
  • int good():操作正常时返回非0值,当操作出错、失败或者读到文件结束符时均为不正常,则返回0。
  • int precision():返回浮点数输出精度,即输出的有效数字的位数。
  • int precision(int n):设置浮点数的输出精度为n,返回此前的输出精度。系统预设置的输出精度为6,即输出的浮点数最多具有6位有效数字。
  • int rdstate():操作正常是返回0,否则返回非0值,它与good正好相反。
  • long setf(long f):根据参数f设置相应的格式化标志,返回此前的设置。该参数f所对应的实参为无名枚举类型中枚举常量(又称格式化常量),可以同时使用一个或多个常量,每两个常量之间要用按位或操作符连接。如当需要左对齐输出,并使数值中的字母大写时,则调用改函数的实参为ios::left | ios::uppercase。
  • long unsetf(long f):根据参数f清除相应的格式化标志,返回此前的设置。如要清除此前的左对齐输出设置,恢复缺省的右对齐输出设置,则调用该函数的实参为ios::left。
  • int width():返回当前的输出域宽。若返回数值0则表明没有为刚才输出的数值设置输出域宽,输出域宽是指输出的值再流中所占有的字节数。
  • int width(int w):设置下一个数据值的输出域宽为w,返回为输出上一个数据值所规定的域宽,若无规定则返回0。注意:此设置不是一直有效,而是只对下一个输出数据有效。

   下面的demo我们姑且认为是方式一

#include <iostream>
using namespace std;
int main()
{
    int x = 30, y = 300, z = 1024;
    cout << x << ' ' << y << ' ' << z << endl;
    cout.setf(ios::oct); // 设置位8进制输出
    cout << x << ' ' << y << ' ' << z << endl;
    cout.unsetf(ios::oct);
    cout.setf(ios::hex);
    cout << x << ' ' << y << ' ' << z << endl;
cout.setf(ios::showbase | ios::uppercase);   
cout << x << ' ' << y << ' ' << z << endl;
    cout.unsetf(ios::showbase | ios::uppercase);
    cout << x << ' ' << y << ' ' << z << endl;
    cout.unsetf(ios::hex);
    cout << x << ' ' << y << ' ' << z << endl;
    return 0;
}

       以上代码可能会运行错误,存在调不出库的情况,本人使用的mingw32编译器,具体细节请查看:

     《关于 QtCreartor编写纯C++程序调用不到C++某些标准库和枚举以及运行错误的解决方法》:

       http://blog.csdn.net/qq21497936/article/details/78983051

现在已改为msvc2015,调用没有问题,运行时,输出结果如下图,与预期不同。

图片.png

原因:

       除非知道当前没有设置基标志,否则 ios::setf(_IFlags) 不应和 ios::dec、ios::oct 或 ios::hex 的标志值一起使用。格式化的输入/输出函数和运算符假定只设置了一个基。改用 ios_base。例如,setf( ios_base::oct, ios_base::basefield ) 清除所有基信息并将基设置成八进制。

程序改为方式二:

#include <iostream>
using namespace std;
int main()
{
    int x = 30, y = 300, z = 1024;
    cout << x << ' ' << y << ' ' << z << endl;
#if 0
    cout.setf(ios::oct); // 设置位8进制输出
#else
    cout.setf(ios::showbase); // 显示进制前缀 八进制36显示为036
    // 清除所有基信息并将基设置成八进制,必须最开始先调用一次ios_base::basefield,
    // 后续单独设置ios_base::oct等才会生效
    cout.setf(ios_base::oct, ios_base::basefield);
#endif
    cout << x << ' ' << y << ' ' << z << endl;
#if 0
    cout.unsetf(ios::oct);
    cout.setf(ios::hex);
#else
//    cout.setf(ios_base::hex, ios_base::basefield);
    cout.unsetf(ios_base::oct);
    cout.setf(ios_base::hex);
#endif
    cout << x << ' ' << y << ' ' << z << endl;
#if 0
    cout.setf(ios::showbase | ios::uppercase); // 设置基指示符输出和数值中的字母大写输出
#else
    cout.setf(ios_base::uppercase);
#endif
    cout << x << ' ' << y << ' ' << z << endl;
    cout.setf(ios_base::dec, ios_base::basefield);
    cout << x << ' ' << y << ' ' << z << endl;
    return 0;
}

运行结果:

图片.png

以下是第三种风格(建议使用方式三替代方式一,不建议使用方式二):

#include <iostream>
using namespace std;
int main()
{
    int x = 30, y = 300, z = 1024;
    cout << x << ' ' << y << ' ' << z << endl;
    cout << oct << x << ' ' << y << ' ' << z << endl;
    cout << hex << x << ' ' << y << ' ' << z << endl;
    cout << showbase << uppercase << x << ' ' << y << ' ' << z << endl;
    cout << dec << x << ' ' << y << ' ' << z << endl;
    return 0;
}

图片.png

2-3格式控制操作符


       数据输入输出的格式更便捷的形式是使用系统头文件<iomanip.h>中提供的操作符,使用这些不需要调用成员函数,只要把他们作为插入操作符<<(个别作为提取操作符>>)的输出对象即可。再成员函数章节中的方式三则是使用该方法。

  • dec,转换为十进制输出整数,系统预制的进制。
  • oct,转换为按八进制输出整数。
  • hex,转换为按十六进制输出整数。
  • ws,从输入流中读取空白字符。
  • endl,输出换行符’\n’并刷新流。(刷新流是指把缓冲区的内容立即写入到对应的物理设备上)。
  • ends,输出一个空字符’\0’。
  • flush,至刷新一个输出流。
  • setiosflags(long f),设置f所对应的格式化标准,功能与setf(longf)成员函数相同。
  • resetiosflags(long f),清楚F所对应的格式化标志,功能与unsetf(longf)成员函数相同。输出后返回流。
  • setfill(int c),设置填充字符位ASCII码为c的字符。
  • setprecision(int n),预置浮点数的输出精度为n。
  • setw(int w),设置下一个数据的输出域宽度位w。

       在上面的操纵符中,dec、oct、hex、endl、ends、flush和ws除了在iomanip.h中有定义外,再iosream.h中也又定义。所以程序中使用这些不带标参数的标记符时,可只包含iostream.h而不需要包含iomanip.h。

       dec、hex和oct这三个函数的作用和c语言的printf中的%d、%x和%o相同,用于输入/输出整数数制的设定。当用户输入时,若输进违例数值,则强制一个零给目标。一旦用上述某个函数设置书之后,在本程序执行中直至下一个设置前,该设置一直有效(已代码验证),且cin与cout不互相影响。

       回车换行函数endl用户产生一个”\n”码,即在输出流中插入0x0d0a内码。所以cout<<endl等效于cout<<’\n’。流操作符ends等于’\0’,该字符主要用于在一组字符后充当字符串结束的标志。

       强制flush主要是为了提高输出效率,在输出时,系统并不会把每个输出项都立即送外外设,而是等待积满某一个缓冲区(如果集装箱)后才一起送出。但用本函数flush即可时正在缓冲区中待输出的内容都被立即输出,同时输出缓存也被清除,此函数多用于流式文件输出当中。

       用取消输入结束符函数ws表示可以省去输入时用作代表一个输入项结束的空格或(Tab)键。这时的输入结束判决完全由对应的变量类型来决定。例如如下代码所示:

char c,d;
cin >> c >> d;

此时的输入:x y<CR>与xy<CR>等效,如下图

图片.png

       如果使用ws函数后,这个效果始终存在,但对数值变量无效。

       输入/输出域宽度设置函数setw(int)用整数参数来指定输入/输出域的宽度。相当于c语言库函数scanf、prinf中的”%”格和格式符的作用。该函数的设置仅其后的一个数据项有效。当用于输出时,若所输出的实际宽度小于设定的数据域宽度,则数据缺省一律向右对齐,反之则按数据的实际宽度输出。当用户输入时,若输入数据的实际宽度超过设定的数据宽度时,超出的数据部分被阶段且被作为下一项输入数据内容。利用此特性可以防止在变量输入时出现的越界情况,但不好用也容易出错。

图片.png

图片.png

图片.png

       向输出域中填充字符的函数setfill(int)常与setw(int)函数联合使用,如下图:

图片.png

       setprecision(int)设置浮点数输出显示精度。

图片.png

       读取/设置域宽的函数int width() constint width(int)

       读取/设置填充字符的函数char fill() constchar fill(char)

       读取/设置浮点数有效位长的函数int precision() constint precision(int)

       以上三组函数 ,必须用发送消息的格式来引用(即以cout.或cin.的形式)。

图片.png

原博主博客地址:http://blog.csdn.net/qq21497936

本文章博客地址:http://mp.blog.csdn.net/postedit/79177645



相关文章
|
14天前
|
机器学习/深度学习 算法 算法框架/工具
为什么使用C++进行机器学习开发
C++作为一种高性能语言,在某些性能要求极高或资源受限的场景下也具有非常重要的地位。C++的高效性和对底层硬件的控制能力,使其在大规模机器学习系统中发挥重要作用,尤其是当需要处理大数据或实时响应的系统时。
29 3
|
2月前
|
算法 C语言 C++
C++语言学习指南:从新手到高手,一文带你领略系统编程的巅峰技艺!
【8月更文挑战第22天】C++由Bjarne Stroustrup于1985年创立,凭借卓越性能与灵活性,在系统编程、游戏开发等领域占据重要地位。它继承了C语言的高效性,并引入面向对象编程,使代码更模块化易管理。C++支持基本语法如变量声明与控制结构;通过`iostream`库实现输入输出;利用类与对象实现面向对象编程;提供模板增强代码复用性;具备异常处理机制确保程序健壮性;C++11引入现代化特性简化编程;标准模板库(STL)支持高效编程;多线程支持利用多核优势。虽然学习曲线陡峭,但掌握后可开启高性能编程大门。随着新标准如C++20的发展,C++持续演进,提供更多开发可能性。
49 0
|
9天前
|
物联网 C# C语言
物联网开发中C、C++和C#哪个更好用
在物联网(IoT)开发中,C、C++和C#各有优缺点,适用场景不同。C语言性能高、资源占用低,适合内存和计算能力有限的嵌入式系统,但开发复杂度高,易出错。C++支持面向对象编程,性能优秀,适用于复杂应用,但学习曲线陡峭,编译时间长。C#易于学习,与.NET框架结合紧密,适合快速开发Windows应用,但性能略低,平台支持有限。选择语言需根据具体项目需求、复杂性和团队技术栈综合考虑。
|
14天前
|
Java Android开发 C++
🚀Android NDK开发实战!Java与C++混合编程,打造极致性能体验!📊
在Android应用开发中,追求卓越性能是不变的主题。本文介绍如何利用Android NDK(Native Development Kit)结合Java与C++进行混合编程,提升应用性能。从环境搭建到JNI接口设计,再到实战示例,全面展示NDK的优势与应用技巧,助你打造高性能应用。通过具体案例,如计算斐波那契数列,详细讲解Java与C++的协作流程,帮助开发者掌握NDK开发精髓,实现高效计算与硬件交互。
54 1
|
4天前
|
C++
HTML+JavaScript构建一个将C/C++定义的ANSI字符串转换为MASM32定义的DWUniCode字符串的工具
HTML+JavaScript构建一个将C/C++定义的ANSI字符串转换为MASM32定义的DWUniCode字符串的工具
|
2月前
|
C++
C++ Qt开发:QUdpSocket网络通信组件
QUdpSocket是Qt网络编程中一个非常有用的组件,它提供了在UDP协议下进行数据发送和接收的能力。通过简单的方法和信号,可以轻松实现基于UDP的网络通信。不过,需要注意的是,UDP协议本身不保证数据的可靠传输,因此在使用QUdpSocket时,可能需要在应用层实现一些机制来保证数据的完整性和顺序,或者选择在适用的场景下使用UDP协议。
75 2
|
20天前
|
JavaScript 前端开发 测试技术
一个google Test文件C++语言案例
这篇文章我们来介绍一下真正的C++语言如何用GTest来实现单元测试。
14 0
|
1月前
|
图形学 C++ C#
Unity插件开发全攻略:从零起步教你用C++扩展游戏功能,解锁Unity新玩法的详细步骤与实战技巧大公开
【8月更文挑战第31天】Unity 是一款功能强大的游戏开发引擎,支持多平台发布并拥有丰富的插件生态系统。本文介绍 Unity 插件开发基础,帮助读者从零开始编写自定义插件以扩展其功能。插件通常用 C++ 编写,通过 Mono C# 运行时调用,需在不同平台上编译。文中详细讲解了开发环境搭建、简单插件编写及在 Unity 中调用的方法,包括创建 C# 封装脚本和处理跨平台问题,助力开发者提升游戏开发效率。
48 0
|
15天前
|
编译器 C++
C++ 类构造函数初始化列表
构造函数初始化列表以一个冒号开始,接着是以逗号分隔的数据成员列表,每个数据成员后面跟一个放在括号中的初始化式。
60 30
|
4天前
|
并行计算 Unix Linux
超级好用的C++实用库之线程基类
超级好用的C++实用库之线程基类
12 4