我有一个std::istream
需要显示为单个std::istream
对象的对象列表。因此,如果我有三个istream
s,A、B 和 C,我希望能够创建一个istream
, D,它将首先返回来自 A 的字节,然后返回来自 B 的字节,然后EOF
到达 C 之前。复合流将始终按顺序读取并在读取所有字节后关闭。
有没有一种使用 stl/boost 的简单方法来做到这一点,还是我只需要编写自己的复合 istream?
另一个问题( 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
实例!