Ответ 1
Я сам был в подобной ситуации, и вместо того, чтобы создавать все, я использовал Boost.Iostreams, создав исходное устройство только для чтения.
Неподтвержденный, но это может работать:
class ConstBufferDevice
{
public:
typedef char char_type;
struct category :
virtual boost::iostreams::device_tag,
virtual boost::iostreams::input_seekable
{
};
ConstBufferDevice(const char_type* buffer, size_t buffersize)
: buffer_(buffer)
, buffersize_(buffersize)
, pos_(0)
{
}
std::streamsize read(char_type* buffer, std::streamsize buffersize)
{
const std::streamsize amount = static_cast<std::streamsize>(buffersize_ - pos_);
const std::streamsize result = (std::min)(buffersize, amount);
if (result != 0)
{
std::copy(buffer_ + pos_, buffer_ + pos_ + result, buffer);
pos_ += result;
return result;
}
else
{
return buffersize ? -1 : 0; // EOF
}
}
std::streampos seek(boost::iostreams::stream_offset offset,
std::ios_base::seekdir seekdir)
{
// Determine new value of pos_
boost::iostreams::stream_offset newpos;
if (seekdir == std::ios_base::beg)
{
newpos = offset;
}
else if (seekdir == std::ios_base::cur)
{
newpos = pos_ + offset;
}
else if (seekdir == std::ios_base::end)
{
newpos = buffersize_ + offset;
}
else
{
throw std::ios_base::failure("bad seek direction");
}
// Check for errors
if (newpos < 0 || newpos > buffersize_)
{
throw std::ios_base::failure("bad seek offset");
}
pos_ = static_cast<size_t>(newpos);
return boost::iostreams::offset_to_position(newpos);
}
private:
const char_type* buffer_;
size_t buffersize_;
size_t pos_;
};
typedef boost::iostreams::stream<ConstBufferDevice> ConstBufferStream;