Obtain a std::ostream either from std::cout or std::ofstream(file)(从 std::cout 或 std::ofstream(file) 获取 std::ostream)
问题描述
如何将 std::ostream 绑定到 std::cout 或 std::ofstream 对象,具体取决于某个程序条件?尽管由于多种原因这无效,但我想实现语义上等同于以下内容的内容:
how do I bind a std::ostream to either std::cout or to an std::ofstream object, depending on a certain program condition? Although this invalid for many reasons, I would like to achieve something that is semantically equivalent to the following:
std::ostream out = condition ? &std::cout : std::ofstream(filename);
我见过一些不是异常安全的例子,例如来自 http://www2.roguewave.com/support/docs/sourcepro/edition9/html/stdlibug/34-2.html:
I've seen some examples that are not exception-safe, such as one from http://www2.roguewave.com/support/docs/sourcepro/edition9/html/stdlibug/34-2.html:
int main(int argc, char *argv[])
{
  std::ostream* fp;                                           //1
  if (argc > 1)
     fp = new std::ofstream(argv[1]);                         //2
  else
     fp = &std::cout                                          //3
  *fp << "Hello world!" << std::endl;                         //4
  if (fp!=&std::cout) 
     delete fp;
}
有人知道更好的异常安全解决方案吗?
Does anyone know a better, exception-safe solution?
推荐答案
std::streambuf * buf;
std::ofstream of;
if(!condition) {
    of.open("file.txt");
    buf = of.rdbuf();
} else {
    buf = std::cout.rdbuf();
}
std::ostream out(buf);
这将 cout 或输出文件流的底层流缓冲与 out 相关联.之后,您可以写入out",它将最终到达正确的目的地.如果您只是希望 std::cout 的所有内容都进入一个文件,您也可以这样做
That associates the underlying streambuf of either cout or the output file stream to out. After that you can write to "out" and it will end up in the right destination. If you just want that everything going to std::cout goes into a file, you can aswell do
std::ofstream file("file.txt");
std::streambuf * old = std::cout.rdbuf(file.rdbuf());
// do here output to std::cout
std::cout.rdbuf(old); // restore
第二种方法的缺点是它不是异常安全的.您可能想编写一个使用 RAII 执行此操作的类:
This second method has the drawback that it's not exception safe. You possibly want to write a class that does this using RAII:
struct opiped {
    opiped(std::streambuf * buf, std::ostream & os)
    :os(os), old_buf(os.rdbuf(buf)) { }
    ~opiped() { os.rdbuf(old_buf); }
    std::ostream& os;
    std::streambuf * old_buf;
};
int main() {
    // or: std::filebuf of; 
    //     of.open("file.txt", std::ios_base::out);
    std::ofstream of("file.txt");
    {
        // or: opiped raii(&of, std::cout);
        opiped raii(of.rdbuf(), std::cout);
        std::cout << "going into file" << std::endl;
    }
    std::cout << "going on screen" << std::endl;
}
现在,无论发生什么,std::cout 都处于干净状态.
Now, whatever happens, std::cout is in clean state.
这篇关于从 std::cout 或 std::ofstream(file) 获取 std::ostream的文章就介绍到这了,希望我们推荐的答案对大家有所帮助,也希望大家多多支持编程学习网!
本文标题为:从 std::cout 或 std::ofstream(file) 获取 std::ostream
 
				
         
 
            
        基础教程推荐
- C++结构和函数声明。为什么它不能编译? 2022-11-07
- 这个宏可以转换成函数吗? 2022-01-01
- 如何通过C程序打开命令提示符Cmd 2022-12-09
- 在 C++ 中计算滚动/移动平均值 2021-01-01
- 如何检查GTK+3.0中的小部件类型? 2022-11-30
- 如何将 std::pair 的排序 std::list 转换为 std::map 2022-01-01
- 我有静态或动态 boost 库吗? 2021-01-01
- 如何在 C++ 中初始化静态常量成员? 2022-01-01
- 静态库、静态链接动态库和动态链接动态库的 .lib 文件里面是什么? 2021-01-01
- 常量变量在标题中不起作用 2021-01-01
 
    	 
    	 
    	 
    	 
    	 
    	 
    	 
    	 
						 
						 
						 
						 
						 
				 
				 
				 
				