获取stdout内容而不禁用cout / cin

时间:2013-09-05 08:35:07

标签: c++ linux terminal console-application

是否可以将终端(先前)写入的内容(stdout)保存到字符串数组中:

vector<string> lines;

在Linux控制台应用程序中没有禁用cin/cout正常功能。

除了字符之外,在终端窗口中还有每个字符的着色信息。有没有办法将它们保存在另一个数组中?

1 个答案:

答案 0 :(得分:3)

如果您的意思是想在程序内写入std::cout并且数据同时出现在终端和某些程序内部数据结构中,您将使用自定义流缓冲区。这里有一个简单的例子:

#include <streambuf>
#include <iostream>

struct capturebuf
    : std::streambuf
{
    std::ostream&   d_out;
    std::streambuf* d_sbuf;
    std::string&       d_str;
public:
    capturebuf(std::ostream& out, std::string& str)
        : d_out(out)
        , d_sbuf(out.rdbuf())
        , d_str(str)
    {
        out.rdbuf(this);
    }
    ~capturebuf() {
        this->d_out.rdbuf(this->d_sbuf);
    }
    int overflow(int c) {
        if (c != std::char_traits<char>::eof()) {
            this->d_str.push_back(c);
        }
        return this->d_sbuf->sputc(c);
    }
    int sync() { return this->d_sbuf->pubsync(); }
};

上面的流缓冲区只会添加写入构造期间引用的字符串的每个字符。字符也会转发到传递的流缓冲区。需要转发到pubsync()以确保在适当的时间刷新基础流。

要实际使用它,您需要将其安装到std::cout。由于capturebuf的构造函数和析构函数已经进行了注册/注销,因此只需构造一个相应的对象:

int main()
{
    std::string capture;
    {
        capturebuf buf(std::cout, capture);
        std::cout << "hello, world\n";
    }
    std::cout << "caught '" << capture << "'\n";
}

这段代码并没有完全按照你的要求进行,但几乎是:你只需要以不同的方式处理换行符。完全缺少的是缓冲,这不是获得功能而是获得性能所需要的。以上内容是在移动设备上输入的,并且可以证明是错综复杂的,但这种方法应该按原样运行。