Как повторно использовать ostringstream?

Я хотел бы очистить и повторно использовать поток ostringstream (и базовый буфер), чтобы моему приложению не приходилось делать столько распределений. Как вернуть объект в исходное состояние?


person twk    schedule 08.03.2009    source источник
comment
возможный дубликат Как в C ++ очистить переменную строкового потока?   -  person mpromonet    schedule 19.10.2014


Ответы (4)


Раньше я использовал последовательность clear и str:

// clear, because eof or other bits may be still set. 
s.clear();
s.str("");

Что сделало это как для входных, так и для выходных строковых потоков. В качестве альтернативы вы можете вручную очистить, а затем с самого начала искать соответствующую последовательность:

s.clear();
s.seekp(0); // for outputs: seek put ptr to start
s.seekg(0); // for inputs: seek get ptr to start

Это предотвратит некоторые перераспределения, выполняемые str, путем перезаписи всего, что находится в выходном буфере в данный момент. Результаты такие:

std::ostringstream s;
s << "hello";
s.seekp(0);
s << "b";
assert(s.str() == "bello");

Если вы хотите использовать строку для c-функций, вы можете использовать std::ends, поместив завершающий нуль следующим образом:

std::ostringstream s;
s << "hello";
s.seekp(0);
s << "b" << std::ends;
assert(s.str().size() == 5 && std::strlen(s.str().data()) == 1);

std::ends - это пережиток устаревшего std::strstream, который мог записывать непосредственно в массив символов, который вы разместили в стеке. Вам пришлось вручную вставить завершающий нуль. Однако std::ends не является устаревшим, я думаю, потому что он по-прежнему полезен, как и в приведенных выше случаях.

person Johannes Schaub - litb    schedule 08.03.2009
comment
Я пытаюсь использовать s.str () с ostream. Размер портит (я вижу, что первый символ нулевой, но он печатает гораздо больше). Есть ли хороший способ исправить длину str? я использую s.str (). c_str (); Банкомат, и он отлично работает - person ; 06.06.2011
comment
На самом деле даже это неверно. Я просто сделал s.str(""); вместо этого. auto str = s.str(); auto cstr = str.c_str(); file << cstr; s.clear(); s.seekp(0); s << ends; - person ; 06.06.2011
comment
std ::nds не работает для меня в тесте Google boost::any a = 1; std::ostringstream buffer; buffer << a << std::ends; EXPECT_EQ( buffer.str(), "any<(int)1>" ); TestUtilsTest.cpp:27: Failure Expected: buffer.str() Which is: "any<(int)1>\0" To be equal to: "any<(int)1>", и если я повторно использую строки разной длины, у меня останутся биты - person David van Laatum; 27.06.2017
comment
Альтернатива - это верный ответ, если вы хотите избежать перераспределения. И если вы действительно хотите начать все заново без перераспределения, просто снова вызовите seekp (0) после отправки std :: end. s.seekp(0); s << std::ends; s.seekp(0); - person Chip Grandits; 30.06.2017

Похоже, что вызов ostr.str("") делает свое дело.

person Diego Sevilla    schedule 08.03.2009
comment
Стоит отметить, что при этом не будет повторно использоваться нижележащий буфер из потока ostring - он просто назначает новый буфер. Итак, пока вы повторно используете объект ostringstream, вы по-прежнему выделяете два буфера. Я не думаю, что ostringstream предназначен для повторного использования в соответствии с вашими намерениями. - person razlebe; 09.03.2009
comment
Он также не очищает состояние, что и делает .clear (). Я согласен, на самом деле он не предназначен для такого использования. Просто создайте новый, чтобы быть уверенным. Только если вы профилируете, вы узнаете, имеет ли это значение. - person Brian Neal; 09.03.2009
comment
sgreeve, Брайан, верно. Обратите внимание, однако, на то, что приведенный выше метод litb требует использования std ::nds. Он повторно использует буфер, но заставляет вас кодировать иначе, как обычно, со строковыми потоками (обычно вы не используете std ::nds). - person Diego Sevilla; 09.03.2009

Если вы собираетесь очистить буфер таким образом, чтобы он был очищен перед первым использованием, вам нужно сначала добавить что-то в буфер с помощью MSVC.

struct Foo {
    std::ostringstream d_str;
    Foo() { 
        d_str << std::ends;   // Add this
    }
    void StrFunc(const char *);
    template<class T>
    inline void StrIt(const T &value) {
        d_str.clear();
        d_str.seekp(0);  // Or else you'll get an error with this seek
        d_str << value << std::ends;
        StrFunc(d_str.str().c_str());  // And your string will be empty
    }
};
person Unkle George    schedule 03.11.2011
comment
Я не вижу сбоев в работе VS2012. Кроме того, вызов clear вызовет failbit устанавливается, если поток пуст. При простом вызове seekp должен просто вернуться, если поток не существует. - person Jonathan Mee; 08.07.2014

Вы этого не сделаете. Для ясности используйте два потока с разными именами и дайте оптимизирующему компилятору понять, что он может повторно использовать старый.

person Sebastian Ganslandt    schedule 08.03.2009
comment
рассмотрим вариант использования, когда код перебирает входные данные, записывает в ostringstream (на основе прочитанных данных), а затем должен записывать строку, построенную в ostringstream где-то время от времени (например, после того, как определенная последовательность символов была прочитана) и начните строить новую строку. - person Andre Holzner; 31.01.2012