《C和C++代码精粹》——1.9 格式化

1.9 格式化

C和C++代码精粹
在程序清单1.8中ios::skipws是一个格式化标志的例子。格式化标志是位掩码值,该位掩码值可以通过成员函数setf来设置,也可用unsetf复位(见表1.1的完整描述)。

程序清单1.9的程序阐述了数字的格式化。标准流成员函数precision用来指定浮点值显示的小数位数。如果没有设置ios::showpoint标志,那么末尾的零不被显示。要用前置加号来打印正数,就用ios::showpos。在上例中想要以16进制形式显示x 和在指数形式中显示大写e,使用ios::uppercase。

程序清单1.7 计算文本文件中的字数

// wc.cpp:显示字的个数
#include <iostream>
using namespace std;  

main()
{
    const size_t BUFSIZ = 128;
    char s[BUFSIZ];
    size_t wc = 0;  

    while (cin >> s)
        ++wc;
    cout << wc << '\n';
}  

//从"wc < wc.cpp”命令输出
34

程序清单1.8 与程序copy1.cpp完全相同,但使用提取运算符读取空格

// copy3.cpp :用>>读取空格符
#include <iostream>
using namespace std;  

main()
{
    char c;  

    //不要跳过空格符
    cin.unsetf(ios::skipws);  

    while (cin >> c)
        cout << c;
}

表1.1 格式化标志

一些格式化选项可以具有一定范围的值。例如,用来确定显示整型数基数的ios::basefield可以被设置成10进制、8进制或16进制。(见表1.2中3种格式化域有效的描述)由于这些是位域而不是单个的位,可用带两个参数形式的setf来设置。例如,程序清单1.10的程序设置8进制数模式采用下面语句:

cout.setf ( ios::oct,ios::basefield );
用标志ios::showbase进行设置时,8进制以0开头,16进制以0x开头打印输出(或者以0X开头打印输出,如果ios::uppercase也被设置)。

程序清单1.9 描述数据格式化

// float.cpp :格式化真正的数字
#include <iostream>
using namespace std;  

main()
{
    float x = 12345.6789, y = 12345;
    cout << x << ' ' << y << '\n';  

    //显示两个十位数
    cout.precision(2);
    cout << x << ' ' << y << '\n';  

    //显示末尾的零
    cout.setf(ios::showpoint);
    cout << x << ' ' << y << '\n';  

    //显示符号
    cout.setf(ios::showpos);
    cout << x << ' ' << y << '\n';  

    //返回符号和默认值的精度
    cout.unsetf(ios::showpos);
    cout.precision(0);  

//使用科学计数法
    cout.setf(ios::scientific,ios::floatfield);
    float z = 1234567890.123456;
    cout << z << '\n';
    cout.setf(ios::uppercase);
    cout << z << '\n';
}  

//输出:
12345.678711 12345
12345.68 12345
12345.68 12345.00
+12345.68 +12345.00
1.234568e+09
1.234568E+09

表1.2 格式化域

程序清单1.10 显示整数的基数

// base1.cpp :显示整数的基数
#include <iostream>
using namespace std;  

main()
{
    int x, ![image](https://yqfile.alicdn.com/d4dff4359c675def6e527db1695916fa57198610.png)
y, z;  

    cout << "输入三个整数: ";
    cin >> x >> y >> z;
    cout << x << ',' << y << ',' << z << endl;  

    //在不同基数中打印
    cout << x << ',';
    cout.setf(ios::oct,ios::basefield);
    cout << y << ',';
    cout.setf(ios::hex,ios::basefield);
    cout << z << endl;  

//显示基数前缀
    cout.setf(ios::showbase);
    cout << x << ',';
    cout.setf(ios::oct,ios::basefield);
    cout << y << ',';
    cout.setf(ios::hex,ios::basefield);
    cout << z << endl;
}  

//运行结果
输入三个整数:10 010 0x10
10,8,16
10,10,10
0xa,010,0x10

本文仅用于学习和交流目的,不代表异步社区观点。非商业转载请注明作译者、出处,并保留本文的原始链接。

时间: 2024-07-31 23:17:53

《C和C++代码精粹》——1.9 格式化的相关文章

《C和C++代码精粹》导读

前言 C和C++代码精粹 本书适合于那些C和C++的职业程序员.假如你已熟悉这两种语言的语法和基本结构,这本书能够为你创建有效的.实用的程序提供实践性的指导.每一个代码范例或程序范例均标明行之有效的用法和技术,这些用法和技术对C/C++这两种重要编程语言的性能发挥起着重要的作用. 对于那些希望在工作中加强自身技术和提高效率的人来说,本书可以算是一本经验之谈.尽管目前人们对面向对象模式的推崇到了白热状态(本书也包括这方面的丰富内容),可是我没有理由不对C++的基础-C表示尊崇.我发现太多的程序开发

《C和C++代码精粹》——1.12 运算符重载

1.12 运算符重载 C和C++代码精粹在C++中你可以重载运算符,例如,定义一个复数的数据类型如下: struct complex { double real, imag; }; 假如能使用中缀符号用于复数加法,那将会相当方便.如: complex c1,c2; - complex c3=c1+c2; 当编译器遇到如c1+c2这样的表达式时,将查找下边两个函数中的一个(只须其中的一个存在): operator+(const complex&,const complex &); //全局函

《C和C++代码精粹》——1.8 标准流

1.8 标准流 C和C++代码精粹C++中有4个预定义的流:cin(标准输入),cout(标准输出),cerr(标准错误),clog(标准错误).除了cerr外其余都是全缓冲流.就像stderr一样,cerr的行为好象是非缓冲的,但事实上它是单元缓冲的,也就是说它在处理完每一个对象而不是每一个字节后会自动清除缓冲.例如,带有单元缓冲的语句: cerr<<"hello":缓冲处理5个字符,然后清除缓冲区.一个非缓冲处理的流会立即发送每个字符到它的最终目的地. 程序清单1.5

《C和C++代码精粹》——2.11 更高深的内容

2.11 更高深的内容 C和C++代码精粹 我们可以很自然地得出以下结论:一个三维数组是二维数组的集合. int a[2] [3] [4]={{{0,1,2,3},{4,5,6,7},{8,9,0,1}}, {{2,3,4,5},{6,7,8,9},{0,1,2,3}}}; 这个数组的第一个元素是一个"二维数组" a[0](从技术上来说,a[0]是一个由3个含有4个整数的数组的数组),为了使指针与数组名a一致,定义: int (*p) [3] [4] = a; 程序清单2.16是一个应

《C和C++代码精粹》——2.7 指针和一维数组

2.7 指针和一维数组 C和C++代码精粹 在程序清单2.7中,会注意到在传递数组 s 时并没有使用它的地址,这是因为C和C++在大多数表达式中把数组名转换成指向它第一个元素的指针.自1984年以来,我已经向成百上千的学生讲授了C和C++,我注意到了指针和数组,特别是指针和多维数组之间的关系造成很多迷惑. 这样说似乎很奇怪,但是C++确实不支持数组,至少C++不像支持第一类数据类型如整型或者甚至结构体那样支持数组.考虑以下的语句: int i=1,j; int a[4]={0,1,2,3},b[

《C和C++代码精粹》——1.10 操纵器

1.10 操纵器 C和C++代码精粹 当标识符 endl出现在一个输出流中时,一个换行字符就被插入并且流被刷新.标识符endl是操纵器的一个例子,即为了副效应而插入到流的一个对象.在〈iostream〉中被声明的系统预定义的操纵器列于表 1.3中.程序清单1.11里的程序在功能上与程序清单1.10的程序等价,但它是用操纵器来代替显式调用setf函数.操纵器经常可以使代码更为高效. 表1.3 简单的操纵器(〈iostream〉) 程序清单1.11 用操纵器改变数据基数 // base2.cpp:

《C和C++代码精粹》——2.5 普通指针

2.5 普通指针 C和C++代码精粹 通常编写能接收指向任意类型参数的函数是很方便的.这是很有必要的,例如,用标准的库函数memcpy,能够从一个地址向另一个地址拷贝一块内存.你也可能想调用memcpy来拷贝自己创建的结构: struct mystruct a,b; /.../ memcpy(&a,&b,sizeof(struct mystruct)); 为了操作任意类型的指针,memcpy把它头两个参数声明为void型指针.可以不需要强制类型转换将任何类型的指针赋予void类型.也可以在

《C和C++代码精粹》——1.2 循序渐进

1.2 循序渐进 C和C++代码精粹 在没有完全掌握C++的情况下也可以有效地使用它.事实上,面向对象技术承诺如果供应商为重用.可扩展性提供设计好的类库,那么建立应用程序的工作就很容易了.现有的开发环境,及其应用程序框架和可视化组件,正在兑现这一承诺. 如果觉得必须要掌握这种语言,可以一步步地去做,并且在这一过程中可以取得丰硕的成果.已出现的3个"顶峰"是: 1.更好的C: 2.数据抽象: 3.面向对象编程. 由于C++比C更安全.更富于表达,所以可以将它作为一个更好的C使用.这个顶峰

《C和C++代码精粹》——2.12 指向函数的指针

2.12 指向函数的指针 C和C++代码精粹 一个指针可以指向函数也可以指向存储的对象.下面的语句声明fp是一个指向返回值为整型(int)的函数的指针: int(*fp)( ); *ftp的圆括号是必需的,没有它的语句 int *fp( ); 将fp声明为一个返回指向整型(int)指针的函数.这就是将星号与类型声明紧密相连的方式成为逐渐受人们欢迎的方式的原因之一. int fp(); //方式说明fp()返回一个指向整型的指针(int ) 当然,这种方式建议你通常应该每条语句只声明一个实体,否则