如何重复使用一个 ostringstream?

我想清除和重用一个 ostringstream (和底层缓冲区) ,这样我的应用程序就不必做那么多的分配。如何将对象重置为初始状态?

58028 次浏览

看来 ostr.str("")呼叫起作用了。

你不知道。为了清晰起见,使用两个不同名称的数据流,让编译器最佳化发现它可以重用旧的数据流。

我过去用过 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-function 使用字符串,你可以使用 std::ends,像这样输入一个结束 null:

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的遗留物,它能够直接写入您在堆栈上分配的字符数组。您必须手动插入终止 null。然而,我认为 std::ends并没有被废弃,因为它在上面的例子中仍然很有用。

如果您要清除缓冲区,使其在首次使用之前被清除,那么您需要首先向缓冲区添加一些内容 w/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
}
};