Как скопировать двоичные данные в поток строк - PullRequest
11 голосов
/ 01 февраля 2011

У меня есть std::vector<int>, и я хочу сериализовать его. Для этого я пытаюсь использовать std::stringstream

 vector<int> v;
 v.resize(10);
 for (int i=0;i<10;i++)
 v[i]=i;


 stringstream ss (stringstream::in | stringstream::out |stringstream::binary);

Однако, когда я копирую вектор в поток строки, он копируется как символ

ostream_iterator<int> it(ss);
copy(v.begin(),v.end(),it);

значение, которое вставлено в буфер (_Strbuf), равно "123456789"

Я предложил написать обходное решение

for (int i=1;i<10;i++)
   ss.write((char*)&p[i],sizeof(int));

Я хочу сделать это как первый способ, используя функцию std, такую ​​как copy

спасибо Герцлю

Ответы [ 4 ]

10 голосов
/ 01 февраля 2011

На самом деле, это ваш обходной путь, но его можно использовать с алгоритмом std :: copy ().

   template<class T>
   struct serialize
   {
      serialize(const T & i_value) : value(i_value) {}
      T value;
   };

   template<class T>
   ostream& operator <<(ostream &os, const serialize<T> & obj)
   {
      os.write((char*)&obj.value,sizeof(T));
      return os;
   }

Использование

ostream_iterator<serialize<int> > it(ss);
copy(v.begin(),v.end(),it);
2 голосов
/ 01 февраля 2011

Я знаю, что это не решение вашей проблемы, но если вы не ограничены STL, вы можете попробовать ( повысить сериализацию ) или буферы протокола Google

Boost даже имеет встроенную поддержку де-сериализации контейнеров STL (http://www.boost.org/doc/libs/1_45_0/libs/serialization/doc/tutorial.html#stl)

1 голос
/ 01 февраля 2011

Как и Фред, я не вижу в этом смысла, то, что вы фактически пытаетесь сделать, это:

ss.rdbuf()->sputn(reinterpret_cast<char*>(&v[0]), sizeof(int) * v.size());
1 голос
/ 01 февраля 2011

Чтобы использовать std :: copy с ostream :: write, вам нужно написать собственный выходной итератор, который знает, как правильно сериализовать тип.Тем не менее, я не уверен, что вы ожидаете получить от этого подхода, но вот первый проход в этом примере:

struct ostream_write_int
  : std::iterator<std::output_iterator_tag, int, void, void, void>
{
  std::ostream *s;
  ostream_write_int(std::ostream &s) : s (&s) {}

  ostream_write_int& operator++() { return *this; }
  ostream_write_int& operator++(int) { return *this; }
  ostream_write_int& operator*() { return *this; }

  void operator=(int x) {
    s->write(reinterpret_cast<char*>(&x), sizeof(x));
  }
};

Это может быть шаблонно, только если выперенести логику сериализации в какую-то другую функцию (как это делают итераторы отформатированного потока для оператора <<). </p>

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...