C++标准输出流换向(cout.rdbuf()打印到文件)
1.原由
cpp复习中学到了I/O流,然后看郑莉老师视频的时候提到了一个标准输出流换向问题。回想当年写操作系统课设的时候要将print的日志写到txt文件中,java不精通的我竟然将一条条print输出到GUI界面上,然后将GUI的日志写入文件。
实在是繁琐,但是cpp中有io流的概念可以采用重定向的方式直接将cout的标准输出重定向到文件中,非常nice,记录下来,供以后使用。(脑子太笨记不住)
2.关键知识
先直接上代码
#include<iostream>
#include<fstream>
using namespace std;
//标准输出流换向
int main(){
ofstream fout("b.txt"); //文件输出流对象
streambuf* pOld =cout.rdbuf(fout.rdbuf());
cout<<"我是sb"<<endl;//将在目录下的b.txt中写入 我是sb
// cout.rdbuf(pOld); //用于恢复 标准输出
}
这里主要用到的就是rdbuf() 函数,但是开始我很困惑,在main第二行的代码中,为什么重定向之后的指针命名是old,而且cout.rdbuf(pOld)恢复之后,还真就能打印到屏幕上,其实主要就是对rdbuf() 函数的使用和返回值不太了解。
rdbuf()函数的声明如下
streambuf* rdbuf ( ) const; //get
streambuf* rdbuf ( streambuf* sb );//set
将第一行的无参函数调用理解为get,作用是返回与流关联的流缓冲区对象,简单理解就是返回我原来输出到哪个缓存区。
将第二行的有参函数调用理解为set,作用是将流与sb指针关联 and 返回先前与流关联的流缓冲区对象,也就是将流输出重定向到sb指针指向的缓存区,同时返回原来指向的缓存区。
那么我们现在再来看这段代码就明了了。
int main(){
ofstream fout("b.txt"); //文件输出流对象
streambuf* pOld =cout.rdbuf(fout.rdbuf());
cout<<"我是sb"<<endl;//将在目录下的b.txt中写入 我是sb
// cout.rdbuf(pOld); //用于恢复 标准输出
}
fout.rdbuf()相当于一个get,返回的是文件输出流的buffer地址,cout.rdbuf(fout.rdout())==cout.rdbuf(get) ——》set作用。也就是将cout的输出重定向到fout那儿,也就是文件中,此时cout标准输出流就重定向到了文件中了。
但是这儿调用第二个函数也就是set还会返回一个cout原来输出的buffer地址,也就是原来的标准输出地址,所以将这个返回值保存到pOld里面,用于恢复标准输出。(此时的pOld就是cout原来标准输出的地址)。
好吧,为了我这个憨憨以后能看懂,再详细一点。
上面的代码,拆分开写就是下面这样。
int main(){
ofstream fout("b.txt"); //文件输出流对象
streambuf* pOld = cout.rdbuf();//get old保存cout的标准输出
streambuf* fi = fout.rdbuf();//get 文件的输出buf
cout.rdbuf(fi);//set 将cout输出buf设置为文件的输出buf 实现重定向
cout<<"我是憨憨"<<endl;
return 0;
}
ok 没问题。
3.结果演示
首先是将cout定向到文件中结果。
int main(){
ofstream fout("b.txt"); //文件输出流对象
streambuf* pOld =cout.rdbuf(fout.rdbuf());
cout<<"我是sb"<<endl;//将在目录下的b.txt中写入 我是sb
}
结果如下:
然后就是还原标准输出结果。
int main(){
ofstream fout("b.txt"); //文件输出流对象
streambuf* pOld =cout.rdbuf(fout.rdbuf());
cout.rdbuf(pOld); //用于恢复 标准输出
cout<<"我是sb"<<endl;
}
可以看到输出已经回到了标准输出,也就是屏幕上了。
over,继续肝!