2

我有一个std::istream需要显示为单个std::istream对象的对象列表。因此,如果我有三个istreams,A、B 和 C,我希望能够创建一个istream, D,它将首先返回来自 A 的字节,然后返回来自 B 的字节,然后EOF到达 C 之前。复合流将始终按顺序读取并在读取所有字节后关闭。

有没有一种使用 stl/boost 的简单方法来做到这一点,还是我只需要编写自己的复合 istream?

4

1 回答 1

1

另一个问题( https://stackoverflow.com/a/17103292/1424877 )的题外话答案可能会对您有所帮助。

#include <iostream>
#include <string>
#include <sstream>

class ConcatStreams : public std::streambuf
{
    int useBuf;
    std::streambuf *sbuf_[2];
    char buffer_[1024];

  public:
    ConcatStreams(std::istream& sbuf1, std::istream& sbuf2) :
        useBuf(0), sbuf_{sbuf1.rdbuf(), sbuf2.rdbuf()}
    { }

    int underflow()
    {
        if (this->gptr() == this->egptr()) {
            std::streamsize size = 0;
            while (useBuf < 2) {
                size = this->sbuf_[useBuf]->sgetn(this->buffer_, sizeof this->buffer_);
                if (!size) {
                    useBuf++;
                } else {
                    break;
                }
            } 
            this->setg(this->buffer_, this->buffer_, this->buffer_ + size);
        }
        return this->gptr() == this->egptr()
             ? std::char_traits<char>::eof()
             : std::char_traits<char>::to_int_type(*this->gptr());
    }
};

int main()
{
    std::istringstream is("hello world!\n");

    ConcatStreams cs_(is, std::cin); // prepend "hello world" to the input
    std::istream cs(&cs_);

    std::string s;
    while (cs >> s)
      std::cout << "'" << s << "'" << std::endl;
}

请注意,您不能使用此特定技巧std::cin与自身连接,甚至is与自身连接;但它应该可以很好地连接任何两个不同的输入流。您甚至可以连接多个ConcatStreams实例!

于 2013-10-22T22:48:54.887 回答