浅谈C++标准IO对象
(本文是本人阅读《C++ Primer 中文第五版》时的一些笔记!)
C++标准库定义了四个IO对象:cin、cout、cerr和clog;其中cin是标准输入(standard input)对象,cout是标准输出(standard output)对象,cerr是标准错误(standard error)对象而clog则是日志输出对象(PS:书中并没有将clog描述为标准日志或者标准日志对象之类的语句,不过我个人就是把它当做标准日志对象!)
IO对象代码示例
以下代码的功能是从标准输入读入一个字符,然后用标准输出、标准错误、标准日志对象将该字符输出!
#include <iostream>
using namespace std;
int main()
{
char ch;
cin >> ch;
cout << "standard output:" << ch << endl;
cerr << "standard error:" << ch << endl;
clog << "standard log:" << ch << endl;
return 0;
}
上述代码的运行结果如下:
2
standard output:2
standard error:2
standard log:2
Process returned 0 (0x0) execution time : 3.518 s
Press any key to continue.
其中2是输入的字符,接下来的三行是输出结果!其它的是codeblock(17.12)输出的运行信息以及操作提示!(PS:codeblock的编辑器中,clog居然没有高亮显示!)。
从上面的运行结果来看,cout、cerr和clog都是向屏幕输出数据!好像没有区别,但是cerr输出时,是没有经过缓冲的,信息直接输出到屏幕,而cout和clog是经过缓冲区的。个人认为,cout和clog的功能是一样的,之所以存在cout和clog着两个标准输出对象,我个人比较认同网上“是为了程序的可读性”这一观点!而cerr对象的存在,网上有一种比较高深的说法:
另外,cin、cout和clog都可以被重定向,而cerr不能被重定向。(cer也是可以被重定向的)
重定向标准IO
重定向标准IO有两种方法:外部重定向和内部重定向(PS:不知道这两个词组是不是专业术语)。
外部重定向
将下面程序(main.cpp)编译为main.exe
#include <iostream>
using namespace std;
int main()
{
char ch;
cin >> ch;
cout << "standard output:" << ch << endl;
cerr << "standard error:" << ch << endl;
clog << "standard log:" << ch << endl;
return 0;
}
利用命令行启动main.exe并重定向标准输出,结果如下图:
首先,我们来看看上图中的终端输入输出部分(用的是git终端)。进入程序所在目录后,执行以下语句:然后输入一个2。
./main.exe >1.txt
按下回车后,我们可以看到,终端输出了:
standard error:2
standard log:2
而代码中的标准输出语句
cout << "standard output:" << ch << endl;
并没有在终端输出任何内容,但是在main.exe所在的目录下生成了一个1.txt的文本文件,我们在图1中可以开到,1.txt的文本内容为:
standard output:2
原来本应在终端输出的内容被输出保存在1.txt的文件中,这就是所谓的重定向标准输出!
同样的道理,标准错误也是可以被重定向的:
从运行结果可以看出,cerr也是可以被重定向的,所以前面我说的cerr不能被重定向是错误的!此外,clog的输出也被重定向到2.txt中,大家可以看到,图2中的命令行用到了两个重定向 >1.txt和2>2.txt。上面的语句可以写成这样:
./main.exe 1>1.txt 2>2.txt
其中1代表的事标准输出(1是可以省略的),2代表的是标出错误。另外0代表的事标准输入。
既然输入能被重定向,那么,输入能被重定向吗?答案是肯定的,直接上一张图:
我们可以看到,直接执行如下命令后,并不需要再次从终端输出数据,cout、cerr和clog都输出了s,而s是1.txt文本内容的首字符。
./main.exe 0<1.txt
上述命令行可以描述为将1.txt当做标准输入,所以,程序重标准输入读取一个字符是,就是冲文本1.txt读入一个字符。
0、1和2到底是什么
0、1和2是文件描述符,0代表标准输入、1代表标准输出、2代表标准错误。
内部重定向
(略)