在网上查到了一些格式化输入输出的文章,这里总结一下。
在本文中参考到的文章链接:
http://c.biancheng.net/cpp/biancheng/view/116.html
http://c.biancheng.net/cpp/biancheng/view/2227.html
前言
在输出数据时,为简便起见,往往不指定输出的格式,由系统根据数据的类型采取默认的格式,但有时希望数据按指定的格式输出。
有两种方法可以达到此目的:
1.使用控制符;
2.使用流对象的有关成员函数。
分别叙述如下
1. 使用控制符
输入和输出并不是C++语言中的正式组成成分。C和C++本身都没有为输入和输出提供专门的语句结构。输入输出不是由C++本身定义的,而是在编译系统提供的I/O库中定义的。C++的输出和输入使用“流”(stream)的方式实现。
有关流对象cin、cout和流运算符的定义等信息是存放在C++的输入输出流库中的,因此如果在程序中使用cin、cout和流运算符,就必须使用预处理命令把头文件stream包含到本文件中:
#include <iostream>
尽管cin和cout不是C++本身提供的语句,但是在不致混淆的情况下,为了叙述方便,常常把由cin和流提取运算符“>>”实现输入的语句称为输入语句或cin语句,把由cout和流插入运算符“<<”实现输出的语句称为输出语句或cout语句。根据C++的语法,凡是能实现某种操作而且最后以分号结束的都是语句。
1.1 输入流与输出流的基本操作
cout << 表达式1 << 表达式2 << …… << 表达式n;
cin >> 变量1 >> 变量2 >> …… >> 变量n;
在定义流对象时,系统会在内存中开辟一段缓冲区,用来暂存输入输出流的数据。在执行cout语句时,先把插入的数据顺序存放在输出缓冲区中,直到输出缓冲区满或遇到cout语句中的endl(或’\n’,ends,flush)为止,此时将缓冲区中已有的数据一起输出,并清空缓冲区。输出流中的数据在系统默认的设备(一般为显示器)输出。
一个cout语句可以分写成若干行,也可写成多个cout语句,如:
// 1
cout << "This is a simple C++ program." << endl;
// 2
cout << "This is " //注意行末尾无分号
<< "a C++ "
<< "program."
<< endl; //语句最后有分号
// 3
cout << "This is "; //语句末尾有分号
cout << "a C++ ";
cout << "program.";
cout << endl;
以上3种情况的输出均为
This is a simple C++ program.
注意,不能用一个插入运算符“<<”插入多个输出项,如:
cout << a, b, c; //错误,不能一次插入多项
cout << a + b + c; //正确,这是一个表达式,作为一项
在用cout输出时,用户不必通知计算机按何种类型输出,系统会自动判别输出数据的类型,使输出的数据按相应的类型输出。如已定义a为int型,b为float型,c为char型,则
cout << a << ' ' << b << ' ' << c << endl;
会以下面的形式输出:
4 345.789 a
与cout类似,一个cin语句可以分写成若干行,如:
// 1
cin >> a >> b >> c >> d;
// 2
cin >> a //注意行末尾无分号
>> b //这样写可能看起来清晰些
>> c
>> d;
// 3
cin >> a;
cin >> b;
cin >> c;
cin >> d;
以上3种情况均可以从键盘输入:
1 2 3 4 ↙
也可以分多行输入数据:
1↙
2 3↙
4↙
在用cin输入时,系统也会根据变量的类型从输入流中提取相应长度的字节,如:
char c1, c2;
int a;
float b;
cin >> c1 >> c2 >> a >> b;
如果输入
1234 56.78↙
注意: 34后面应该有空格以便和56.78分隔开。也可以按下面格式输入:
1 2 34 56.78↙ (在1和2之间有空格)
不能用cin语句把空格字符和回车换行符作为字符输入给字符变量,它们将被跳过。
如果想将空格字符或回车换行符(或任何其他键盘上的字符)输入给字符变量,可以使用getchar函数。
1.2 在输入流与输出流中使用控制符
上面介绍的是使用cout和cin时的默认格式。但有时人们在输入输出时有一些特殊的要求,如在输出实数时规定字段宽度,只保留两位小数,数据向左或向右对齐等。C++提供了在输入输出流中使用的控制符(有的书中称为操纵符),见下表:
控制符 | 作用 |
---|---|
dec | 设置数值的基数为10 |
hex | 设置数值的基数为16 |
oct | 设置数值的基数为8 |
setfill© | 设置填充字符c,c可以是字符常量或字符变量 |
setprecision(n) | 设置浮点数的精度为n位。在以一般十进制小数形式输出时,n代表有效数字。在以fixed(固定小数位数)形式和 scientific(指数)形式输出时,n为小数位数 |
setw(n) | 设置字段宽度为n位 |
setiosflags( ios::fixed) | 设置浮点数以固定的小数位数显示 |
setiosftags( ios::scientific) | 设置浮点数以科学记数法(即指数形式)显示 |
setiosflags( ios::left) | 输出数据左对齐 |
setiosflags( ios::right) | 输出数据右对齐 |
setiosflags( ios::skipws) | 忽略前导的空格 |
setiosflags( ios::uppercase) | 数据以十六进制形式输出时字母以大写表示 |
setiosflags( ios::lowercase) | 数据以十六进制形式输出时宇母以小写表示 |
setiosflags(ios::showpos) | 输出正数时给出“+”号 |
defaultfloat | 使用defaultfloat可以关闭所使用的所有set更改 |
需要注意的是: 如果使用了控制符,在程序单位的开头除了要加iostream头文件外,还要加iomanip头文件:
#include <iomanip>
举例, 输出双精度数:
double a = 123.456789012345; // 对a赋初值
cout << a; // 输出: 123.456
cout << setprecision(9) << a; // 输出: 123.456789
cout << setprecision(6); // 恢复默认格式(精度为6)
cout << setiosflags(ios∷fixed); // 输出: 123.456789
cout << setiosflags(ios∷fixed) << setprecision(8) << a; // 输出: 123.45678901
cout << setiosflags(ios∷scientific) << a; // 输出: 1.234568e+02
cout << setiosflags(ios∷scientific) << setprecision(4) << a; // 输出: 1.2346e02
下面是整数输出的例子:
int b = 123456; // 对b赋初值
cout << b; //输出: 123456
cout << hex << b; //输出: 1e240
cout << setiosflags(ios∷uppercase) << b; //输出: 1E240
cout << setw(10) << b << ',' << b; //输出: 123456,123456
cout << setfill('*') << setw(10) << b; //输出: **** 123456
cout << setiosflags(ios∷showpos) << b; //输出: + 123456
如果在多个cout语句中使用相同的setw(n),并使用setiosflags(ios::right),可以实现各行数据右对齐,如果指定相同的精度,可以实现上下小数点对齐。
2. 使用流对象的成员函数
除了可以用控制符来控制输出格式外,还可以通过调用流对象cout中用于控制输出格式的成员函数来控制输出格式。用于控制输出格式的常用的成员函数见下表:
流成员函数 | 作用相同的控制符 | 作用 |
---|---|---|
precision(n) | setprecision(n) | 设置实数的精度为n位 |
width(n) | setw(n) | 设置字段宽度为n位 |
fill(x) | setfill(x) | 设置填充字符x |
setf() | setiosflags() | 设置输出格式状态,括号中应给出格式状态,内容与控制符setiosflags括号中的内容相同 |
unsetf() | resetioflags() | 终止已设置的输出格式状态,在括号中应指定内容 |
流成员函数setf和控制符setiosflags括号中的参数表示格式状态,它是通过格式标志来指定的。格式标志在类ios中被定义为枚举值。因此在引用这些格式标志时要在前面加上类名ios和域运算符“::”。格式标志见下表:
格式标志 | 作用 |
---|---|
ios::left | 输出数据在本域宽范围内向左对齐 |
ios::right | 输出数据在本域宽范围内向右对齐 |
ios::internal | 数值的符号位在域宽内左对齐,数值右对齐,中间由填充字符填充 |
ios::dec | 设置整数的基数为10 |
ios::oct | 设置整数的基数为8 |
ios::hex | 设置整数的基数为16 |
ios::showbase | 强制输出整数的基数(八进制数以0打头,十六进制数以0x打头) |
ios::showpoint | 强制输出浮点数的小点和尾数0 |
ios::uppercase | 在以科学记数法格式E和以十六进制输出字母时以大写表示 |
ios::showpos | 对正数显示“+”号 |
ios::scientific | 浮点数以科学记数法格式输出 |
ios::fixed | 浮点数以定点格式(小数形式)输出 |
ios::unitbuf | 每次输出之后刷新所有的流 |
ios::stdio | 每次输出之后清除stdout, stderr |
#include <iostream>
using namespace std;
int main()
{
int a = 21;
cout.setf(ios::showbase); //显示基数符号(0x或)
cout << "dec:" << a << endl; //默认以十进制形式输出a
cout.unsetf(ios::dec); //终止十进制的格式设置
cout.setf(ios::hex); //设置以十六进制输出的状态
cout << "hex:" << a << endl; //以十六进制形式输出a
cout.unsetf(ios::hex); //终止十六进制的格式设置
cout.setf(ios::oct); //设置以八进制输出的状态
cout << "oct:" << a << endl; //以八进制形式输出a
cout.unseft(ios::oct);
char* pt = "China"; //pt指向字符串"China"
cout.width(10); //指定域宽为
cout << pt << endl; //输出字符串
cout.width(10); //指定域宽为
cout.fill('*'); //指定空白处以'*'填充
cout << pt << endl; //输出字符串
double pi = 22.0 / 7.0; //输出pi值
cout.setf(ios::scientific); //指定用科学记数法输出
cout << "pi="; //输出"pi="
cout.width(14); //指定域宽为
cout << pi << endl; //输出pi值
cout.unsetf(ios::scientific); //终止科学记数法状态
cout.setf(ios::fixed); //指定用定点形式输出
cout.width(12); //指定域宽为
cout.setf(ios::showpos); //正数输出“+”号
cout.setf(ios::internal); //数符出现在左侧
cout.precision(6); //保留位小数
cout << pi << endl; //输出pi,注意数符“+”的位置
return 0;
}
运行情况如下:
dec:21 (十进制形式)
hex:0x15 (十六进制形式,以x开头)
oct:025 (八进制形式,以开头)
China (域宽为)
*****China (域宽为,空白处以'*'填充)
pi=**3.142857e+00 (指数形式输出,域宽,默认位小数)
+***3.142857 (小数形式输出,精度为,最左侧输出数符“+”)
对程序的几点说明:
- 成员函数width(n)和控制符setw(n)只对其后的第一个输出项有效。
但是fill函数在设置后将一直有效,除非被重新设定。这一点与width()十分不同。width只影响他设置后的下一个输出,再下一个字段输出后,后继的字段被恢复为默认值0.
如:
cout.width(6);
cout << 20 << 3.14 << endl;
输出结果为 203.14
在输出第一个输出项20时,域宽为6,因此在20前面有4个空格,在输出3.14时,width (6)已不起作用,此时按系统默认的域宽输出(按数据实际长度输出)。如果要求在输出数据时都按指定的同一域宽n输出,不能只调用一次width(n),而必须在输出每一项前都调用一次width(n>,上面的程序中就是这样做的。
- 输出格式状态分为5组,每一组中同时只能选用一种(例如dec、hex和oct中只能选一,它们是互相排斥的)。在用成员函数setf和控制符setiosflags设置输出格式状态后,如果想改设置为同组的另一状态,应当调用成员函数unsetf(对应于成员函数self)或resetiosflags(对应于控制符setiosflags),先终止原来设置的状态。然后再设置其他状态,大家可以从本程序中看到这点。程序在开始虽然没有用成员函数self和控制符setiosflags设置用dec输出格式状态,但系统默认指定为dec,因此要改变为hex或oct,也应当先用unsetf 函数终止原来设置。如果删去程序中的第7行和第10行,虽然在第8行和第11行中用成员函数setf设置了hex和oct格式,由于未终止dec格式,因此hex和oct的设置均不起作用,系统依然以十进制形式输出。
同理,程序倒数第8行的unsetf 函数的调用也是不可缺少的。 - 用setf 函数设置格式状态时,可以包含两个或多个格式标志,由于这些格式标志在ios类中被定义为枚举值,每一个格式标志以一个二进位代表,因此可以用位或运算符“|”组合多个格式标志。如倒数第5、第6行可以用下面一行代替:
cout.setf(ios::internal I ios::showpos); //包含两个状态标志,用"|"组合
- 可以看到:对输出格式的控制,既可以用控制符,也可以用cout流的有关成员函数,二者的作用是相同的。控制符是在头文件iomanip中定义的,因此用控制符时,必须包含iomanip头文件。cout流的成员函数是在头文件iostream 中定义的,因此只需包含头文件iostream,不必包含iomanip。许多程序人员感到使用控制符方便简单,可以在一个cout输出语句中连续使用多种控制符。