1、除了条件状态之外,每个iostream对象还维持一个控制IO格式化细节的状态。如整形值的基数,浮点值的精度,输出元素的宽度等。标准库还定义了一组操纵符来修改对象的格式状态。操纵符(manipulator)是可用作输入或输出运算符的操作数,它是一个函数或对象(a manipulator is a function or object that can be used as an operand([]操作数;运算对象) to an input or output operator(运算符).)。操纵符返回其应用于的流对象,所以可以在一个语句中输出多个操纵符和数据。

2、操纵符罗列

1 iostream 中定义的操纵符

boolalpha

将真和假显示为字符串

x

noboolalpha

将真和假显示为1,0

showbase

产生指出数的基数的前缀

x

noshowbase

不产生记数基数前缀

showpoint

总是显示小数点

x

noshowpoint

有小数部分才显示小数点

showpos

显示非负数中的+

x

noshowpos

不显示非负数中的+

uppercase

在十六进制中打印0X,科学记数法中打印E

x

nouppercase

在十六进制中打印0x,科学记数法中打印e

x

dec

用十进制显示(整形值,符点数不受影响)

hex

用十六进制显示(整形值,符点数不受影响)

oct

用八进制显示(同上)

left

在值的右边增加填充字符

right

在值的左边增加填充字符

internal

在符号和值之间增加填充字符

fixed

用小数形式显示浮点数

scientific

用科学记数法显示浮点数

boolalpha

将真和假显示为字符串

flush

刷新ostream缓冲区

ends

插入空字符,然后刷新ostream缓冲区

endl

插入换行符,然后刷新ostream缓冲区

unitbuf

在每个输出操作之后刷新缓冲区

x

nounitbuf

恢复常规缓冲区刷新

x

skipws

为输入操作符跳过空白

noskipws

不为输入操作符跳过空白

ws

吃掉空白

注:带 x 的是默认流状态。

2 iomanip 中定义的操纵符

setfill(ch)

ch 填充空白

setprecision(n)

将浮点精度置为 n

setw(w)

读写 w 个字符的值

setbase(b)

按基数 b 输出整数

3、读写操纵符的时候,不读写数据,相反,会采取某种行动。改变流格式状态的操纵符通常为后续IO保留改变后的格式状态。取消操纵符的任何状态改变通常是最好的。一般而言,流应该在每个IO操作之后处于通常的默认状态。大多数改变格式状态的操纵符提供设置/复原对,一个操纵符将格式状态置为新值而另一个进行复原,恢复常规默认格式。

4、用flags操作恢复格式状态

    flags 操作类似于管理流的条件状态的rdstatesetstate操作。这种情况下,标准库定义了一对flags函数:

• 不带实参的 flags() 返回流的当前格式状态。返回值是名为 fmtflags的标准库定义类型。

flags(arg) 接受一个实参并将流格式置为实参所指定的格式。

示例代码

void display(ostream& os)
{
	//remember the current format state
	ostream::fmtflags curr_fmt = os.flags();
	//do output that uses manipulators that change the format state of os
	os.flags(curr_fmt); //restore the original format state of os
}

5、控制输出格式

    有两大类的输出控制:控制数值的表示,以及控制填充符的数据和布局。

示例代码

int main()
{	
	bool bool_val;
	cout << boolalpha
		 << bool_val
		 << noboolalpha;
	return 1;
}

6、控制浮点值的格式

1)对于浮点值的格式化,可以控制下面三个方面:

• 精度:显示多少位数字(总位数)。

• 记数法:用小数还是科学记法法显示。

• 对是整数的浮点值的小数点的处理。

示例代码

int main()
{	
	cout << "Precision: " << cout.precision()
		 << ", value: " << sqrt(2.0) << endl;
	int prev_precision = cout.precision(12);
	cout << ", value: " << sqrt(2.0) << endl;
	cout << setprecision(prev_precision);
	cout << ", value: " << sqrt(2.0) << endl;
	return 1;
}

2)恢复浮点值的默认记数法

    与其他操纵符不同,我们必须调用unsetf成员来取消scientificfixed所做的改变。

示例代码

#include <iostream>
#include "string"
#include "cmath"
using namespace std;

int main()
{	
	cout << sqrt(2.0) << '\n' << endl;
	cout << scientific << sqrt(2.0) << endl;
	cout << fixed << sqrt(2.0) << endl;
	//reset to default handling for notation
	cout.unsetf(ostream::floatfield);
	cout << sqrt(2.0) << endl;
	return 1;
}

7、填充输出

setw,指定下一个数值或字符串的最小间隔。

left,左对齐输出。

right,右对齐输出。输出默认为右对齐。

internal,控制负值的符号位置。internal 左对齐符号且右对齐值,用空格填充介于其间的空间。

setfill,使我们能够指定填充输出时使用的另一个字符。默认情况下,值是空格。

    像 endl 一样,setw 不改变输出流的内部状态,它只决定下一个输出的长度。

示例代码

#include <iostream>
#include "string"
#include "cmath"
#include "iomanip"
using namespace std;

int main()
{	
	int i = -16;
	double d = 3.14159;
	cout << setfill('#')
		 << "i: " << setw(12) << i << endl
		 << "d: " << setw(12) << d << endl
		 << setfill(' '); //restore normal pad character
	return 1;
}

8、默认情况下,输入操作符忽略空白(空格,制表符,换行符,进纸和回车)。noskipws则强制要求不跳过空白。