如何重用ostringstream?

时间:2009-03-08 20:53:24

标签: c++ stl reset ostringstream

我想清除并重用ostringstream(和底层缓冲区),以便我的应用程序不必执行尽可能多的分配。如何将对象重置为初始状态?

4 个答案:

答案 0 :(得分:151)

我过去使用了一系列clear和str:

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

这为输入和输出字符串流做了一件事。或者,您可以手动清除,然后在begin:

中寻找适当的序列
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,像这样放置一个终止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的遗留物,它能够直接写入您在堆栈上分配的char数组。您必须手动插入终止空值。但是,std::ends并未弃用,我认为因为它在上述情况下仍然有用。

答案 1 :(得分:5)

似乎ostr.str("")调用可以解决问题。

答案 2 :(得分:2)

如果您要在第一次使用之前清除缓冲区以使其被清除,您需要首先使用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
    }
};

答案 3 :(得分:0)

你没有。为清晰起见,使用两个不同命名的流,让优化编译器确定它可以重用旧的。