Как читать весь поток в std::string?

Я пытаюсь прочитать весь поток (несколько строк) в строку.

Я использую этот код, и он работает, но он оскорбляет мое чувство стиля... Наверняка, есть более простой способ? Может быть, с помощью stringstream?

void Obj::loadFromStream(std::istream & stream)
{ 
  std::string s;

  std::streampos p = stream.tellg();  // remember where we are

  stream.seekg(0, std::ios_base::end); // go to the end
  std::streamoff sz = stream.tellg() - p;  // work out the size
  stream.seekg(p);        // restore the position

  s.resize(sz);          // resize the string
  stream.read(&s[0], sz);  // and finally, read in the data.


Фактически, ссылка const на строку также будет работать, и это может облегчить задачу...
const std::string &s(... a miracle occurs here...)

Ответ 1

Как насчет

std::istreambuf_iterator<char> eos;
std::string s(std::istreambuf_iterator<char>(stream), eos);

(может быть однострочным, если не для MVP)

post-2011 edit, этот подход теперь пишется

std::string s(std::istreambuf_iterator<char>(stream), {});

Ответ 2

Я опаздываю на вечеринку, но вот довольно эффективное решение:

std::string gulp(std::istream &in)
{
    std::string ret;
    char buffer[4096];
    while (in.read(buffer, sizeof(buffer)))
        ret.append(buffer, sizeof(buffer));
    ret.append(buffer, in.gcount());
    return ret;
}

Я сделал некоторый бенчмаркинг, и выясняется, что метод std::istreambuf_iterator (используемый принятым ответом) на самом деле намного медленнее. В gcc 4.4.5 с -O3 он имеет разницу в 4,5 раза на моей машине, и разрыв становится шире при более низких настройках оптимизации.

Ответ 3

Вы могли бы сделать

std::string s;
std::ostringstream os;
os<<stream.rdbuf();
s=os.str();

но я не знаю, эффективнее ли это.

Альтернативная версия:

std::string s;
std::ostringstream os;
stream>>os.rdbuf();
s=os.str();

Ответ 4

Вы можете попробовать использовать что-то из алгоритмов. Мне нужно подготовиться к работе, но здесь очень быстрый удар по вещам (там должен быть лучший способ):

copy( istreambuf_iterator<char>(stream), istreambuf_iterator<char>(), back_inserter(s) );

Ответ 5

Хорошо, если вы ищете простой и понятный способ сделать это. Я бы рекомендовал добавить/использовать некоторые высокоуровневые рамки для вашего проекта. Для этого я всегда использую Poco и Boost во всех моих проектах. В этом случае с Poco:

    string text;
    FileStream fstream(TEXT_FILE_PATH);
    StreamCopier::copyToString(fstream, text);

Ответ 6

Возможно, это однострочное решение на С++ 11:

std::vector<char> s{std::istreambuf_iterator<char>{in},{}};