web-dev-qa-db-ja.com

ostringstreamを再利用する方法は?

アプリで多くの割り当てを行う必要がないように、ostringstream(および基礎となるバッファー)をクリアして再利用したいと思います。オブジェクトを初期状態にリセットするにはどうすればよいですか?

115
twk

過去に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を使用して、次のように終端の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は非推奨ではありません。上記の場合と同様にまだ有用だと思います。

151

ostr.str("")呼び出しがトリックを行っているようです。

5
Diego Sevilla

最初に使用する前にバッファをクリアする方法でバッファをクリアする場合は、最初に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
    }
};
2
Unkle George

あなたはしません。明確にするために2つの異なる名前のストリームを使用し、最適化コンパイラに古いストリームを再利用できることを理解させます。

0