Я хотел бы очистить и повторно использовать поток ostringstream (и базовый буфер), чтобы моему приложению не приходилось делать столько распределений. Как вернуть объект в исходное состояние?
Как повторно использовать ostringstream?
Ответы (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
не является устаревшим, я думаю, потому что он по-прежнему полезен, как и в приведенных выше случаях.
s.str("");
вместо этого. auto str = s.str(); auto cstr = str.c_str(); file << cstr; s.clear(); s.seekp(0); s << ends;
- person ; 06.06.2011
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
s.seekp(0); s << std::ends; s.seekp(0);
- person Chip Grandits; 30.06.2017
Похоже, что вызов ostr.str("")
делает свое дело.
Если вы собираетесь очистить буфер таким образом, чтобы он был очищен перед первым использованием, вам нужно сначала добавить что-то в буфер с помощью 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
}
};
Вы этого не сделаете. Для ясности используйте два потока с разными именами и дайте оптимизирующему компилятору понять, что он может повторно использовать старый.
ostringstream
(на основе прочитанных данных), а затем должен записывать строку, построенную в ostringstream
где-то время от времени (например, после того, как определенная последовательность символов была прочитана) и начните строить новую строку.
- person Andre Holzner; 31.01.2012