Як повторно використовувати ostringstream?


117

Я хотів би очистити і повторно використати ostringstream (і основний буфер), щоб моїй програмі не довелося робити стільки виділень. Як повернути об'єкт до його початкового стану?


Відповіді:


156

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


Я намагаюся використовувати s.str () з ostream. Розмір заплутав його (я бачу, що перший символ є нульовим, але він друкує набагато більше). Чи є хороший спосіб закріпити довжину стрічки? я використовую s.str (). c_str (); Банкомат, і він працює чудово

Насправді навіть це не правильно. Я просто зробив s.str("");замість цього. auto str = s.str(); auto cstr = str.c_str(); file << cstr; s.clear(); s.seekp(0); s << ends;

std :: end не працює для мене в тесті google, 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>" і якщо я повторно використовую рядки різної довжини, мені залишаються біти
David van Laatum

Альтернатива - це справжня відповідь, якщо ви хочете уникнути перерозподілу. І якщо ви хочете по-справжньому "почати по-новому" без перерозподілу, просто зателефонуйте на пошук searchp (0) знову після відправлення std :: end. s.seekp(0); s << std::ends; s.seekp(0);
Chip Grandits

5

Здається, що ostr.str("")дзвінок виконує трюк.


9
Варто зазначити, що це не повторно використовувати базовий буфер з ostringstream - він просто призначає новий буфер. Тож, використовуючи об'єкт ostringstream, ви все ще виділяєте два буфери. Я не думаю, що ostringstream розроблений для повторного використання способом, який ви плануєте.
разлебе

2
Він також не з'ясовує стан, що робить .clear (). Я погоджуюся, це насправді не призначене для використання таким чином. Просто створіть новий, щоб бути впевненим. Лише за допомогою профілю ви дізнаєтесь, чи має це значення.
Брайан Ніл

1
sgreeve, Брайан, саме так. Зауважте, однак, як вищезазначений метод litb вимагає використання std :: end. Він повторно використовує буфер, але змушує вас кодувати інакше, як зазвичай, за допомогою stringstreams (як правило, ви не використовуєте std :: end).
Дієго Севілья

2

Якщо ви збираєтеся очистити буфер таким чином, що він змусить його очистити до першого використання, вам потрібно буде додати щось до буфера спочатку 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
    }
};

Я не бачу невдалої поведінки на VS2012. Крім того, виклик clearбуде викликати до failbitвстановлювати , якщо потік порожній. Хоча просто дзвінок seekpповинен просто повернутися, якщо потоку немає.
Джонатан Мее

0

Ви цього не робите. Використовуйте два по-різному названі потоки для ясності та дозвольте оптимізуючим компілятору зрозуміти, що він може повторно використовувати старий.


4
розглянемо випадок використання, коли код перебирає вхідні дані, записує на ostringstream(на основі даних, що читаються), а потім повинен записувати рядок, побудований ostringstreamдесь час від часу (наприклад, після того, як прочитана певна послідовність символів), і почати побудова нового рядка.
Андре Хольцнер
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.