我有一个CONST向量,但需要一个函数来读取它。问题是,函数需要一个istream。
将向量中的数据传递给此函数的最有效方法是什么?
我试过这种方法来防止复制流,但它不起作用,因为我的vector是const而pubsetbuf需要一个非const char *。
istringstream is;
const vector<char>& data = GETDATA();
is.rdbuf()->pubsetbuf(&data[0], data.size());
由于流是只读的,我可以转换指针。这样安全吗? 还是更好的方法?
答案 0 :(得分:3)
考虑使用Boost.Iostreams:
#include <boost/iostreams/device/array.hpp>
#include <boost/iostreams/stream.hpp>
using namespace boost::iostreams;
stream<array_source> bs(data.data(), data.size());
std::istream &is = bs;
答案 1 :(得分:2)
你可以写自己的流:
#include <algorithm>
#include <ostream>
// BasicSequenceInputBuffer
// ============================================================================
template < typename Char, typename Traits = std::char_traits<Char> >
class BasicSequenceInputBuffer
: public std::basic_streambuf<Char, Traits>
{
// Types
// =====
private:
typedef std::basic_streambuf<Char, Traits> Base;
public:
typedef typename Base::char_type char_type;
typedef typename Base::int_type int_type;
typedef typename Base::pos_type pos_type;
typedef typename Base::off_type off_type;
typedef typename Base::traits_type traits_type;
typedef const char_type* pointer;
typedef std::size_t size_type;
// Construction
// ============
public:
BasicSequenceInputBuffer(pointer data, size_type size) {
// These casts are safe (no modification will take place):
char* begin = const_cast<char_type*>(data);
char* end = const_cast<char_type*>(data + size);
this->setg(begin, begin, end);
}
// Stream Buffer Interface
// =======================
protected:
virtual std::streamsize showmanyc();
virtual std::streamsize xsgetn(char_type*, std::streamsize);
virtual int_type pbackfail(int_type);
// Utilities
// =========
protected:
int_type eof() { return traits_type::eof(); }
bool is_eof(int_type ch) { return ch == eof(); }
};
// Get Area
// ========
template <typename Char, typename Traits>
std::streamsize
BasicSequenceInputBuffer<Char, Traits>::showmanyc() {
return this->egptr() - this->gptr();
}
template <typename Char, typename Traits>
std::streamsize
BasicSequenceInputBuffer<Char, Traits>::xsgetn(char_type* p, std::streamsize n) {
std::streamsize result = std::min(n, this->egptr() - this->gptr());
std::copy(this->gptr(), this->gptr() + result, p);
this->gbump(result);
return result;
}
template <typename Char, typename Traits>
typename BasicSequenceInputBuffer<Char, Traits>::int_type
BasicSequenceInputBuffer<Char, Traits>::pbackfail(int_type ch) {
if(is_eof(ch)) {
if(this->eback() != this->gptr()) {
this->gbump(-1);
return traits_type::to_int_type(*this->gptr());
}
}
return eof();
}
typedef BasicSequenceInputBuffer<char> SequenceInputBuffer;
// BasicSequenceInputStream
//=============================================================================
template < typename Char, typename Traits = std::char_traits<Char> >
class BasicSequenceInputStream
: public std::basic_istream<Char, Traits>
{
private:
typedef std::basic_istream<Char, Traits> Base;
public:
typedef typename Base::char_type char_type;
typedef typename Base::int_type int_type;
typedef typename Base::pos_type pos_type;
typedef typename Base::off_type off_type;
typedef typename Base::traits_type traits_type;
private:
typedef BasicSequenceInputBuffer<Char, Traits> buffer_type;
public:
typedef typename buffer_type::pointer pointer;
typedef typename buffer_type::size_type size_type;
// Construction
// ============
public:
explicit BasicSequenceInputStream(pointer data, size_type size)
: Base(&m_buf), m_buf(data, size)
{}
private:
buffer_type m_buf;
};
typedef BasicSequenceInputStream<char> SequenceInputStream;
// Test
// ====
#include <iostream>
#include <vector>
int main()
{
const std::string s = "Hello World";
const std::vector<char> vector(s.begin(), s.end());
SequenceInputStream stream(vector.data(), vector.size());
std::string line;
std::getline(stream, line);
std::cout << line << '\n';
}