Как повторно использовать Ostringstream? - PullRequest
115 голосов
/ 08 марта 2009

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

Ответы [ 4 ]

151 голосов
/ 09 марта 2009

Я использовал последовательность 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 не считается устаревшим, я думаю, потому что он все еще полезен, как в вышеупомянутых случаях.

5 голосов
/ 09 марта 2009

Кажется, что вызов ostr.str("") помогает.

2 голосов
/ 03 ноября 2011

Если вы собираетесь очистить буфер таким образом, чтобы он очищался перед первым использованием, вам нужно сначала что-то добавить в буфер с 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
    }
};
0 голосов
/ 09 марта 2009

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

...