Форум программистов, компьютерный форум, киберфорум
С++ для начинающих
Войти
Регистрация
Восстановить пароль
Блоги Сообщество Поиск Заказать работу  
 
Рейтинг 5.00/11: Рейтинг темы: голосов - 11, средняя оценка - 5.00
901 / 478 / 93
Регистрация: 10.06.2014
Сообщений: 2,700

Странное поведение std::string, полученного из std::stringstream

24.03.2018, 21:59. Показов 2325. Ответов 6
Метки нет (Все метки)

Студворк — интернет-сервис помощи студентам
Есть код

C++
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
#include <iostream>
#include <sstream>
#include <cstring>
 
int main()
{
    std::stringstream stream;
    stream << "Hello, world!"
    << 123
    ;
    const char* cstr = stream.str().c_str();
    
    std::cout << "Address saved in variable: " << (void*)cstr << '\n';
    
    std::cout << "Direct reading address: " << (void*)stream.str().c_str() << '\n';
    std::cout << "Direct reading address: " << (void*)stream.str().c_str() << '\n';
    
    std::cout << "Strlen from const char* cstr is: " << strlen(cstr) << '\n';
    std::cout << "Strlen from stream.str().c_str() is: " << strlen(stream.str().c_str());
}
Который выдает следующий результат на GCC
Address saved in variable: 0x1bc3e30
Direct reading address: 0x1bc4e40
Direct reading address: 0x1bc4e40
Strlen from const char* cstr is: 105
Strlen from stream.str().c_str() is: 16
http://rextester.com/ZPHVX79228

Здесь меня смущает следующее:
Почему при получении strlen от переменной в которую был сохранен адрес Си-строки равен 105? А иногда 102, от запуска к запуску значение может меняться, при этом в коде все остается без изменений. Кроме того, длина строки слишком завышена, столько байт в данной строке просто нет.

При этом если взять strlen напрямую от stream.str().c_str() то результат вполне ожидаемый. В чем принципиальное отличие?
Эти странности начинают происходить как только в поток добавляется числовое значение и если его убрать, подобная ситуация не возникает
C++
1
2
3
4
std::stringstream stream;
stream << "Hello, world!"
//<< 123
;
Подобное поведение так же было замечено в Visual Studio Community 2015
0
Лучшие ответы (1)
cpp_developer
Эксперт
20123 / 5690 / 1417
Регистрация: 09.04.2010
Сообщений: 22,546
Блог
24.03.2018, 21:59
Ответы с готовыми решениями:

ошибка error: cannot convert 'std::string {aka std::basic_string<char>}' to 'std::string* {aka std::basic_stri
на вод поступают 2 строки типа string. определить количество вхождений строки 2 в строку 1 ошибка error: cannot convert 'std::string {aka...

На основе исходного std::vector<std::string> содержащего числа, создать std::vector<int> с этими же числами
подскажите есть вот такая задача. Есть список . Создать второй список, в котором будут все эти же числа, но не в виде строк, а в виде...

Странное рекурсивное поведение объекта std::cout
Пытался организовать очередь при помощи шаблона и наткнулся на &quot;странное&quot; поведение: // Организация очереди при помощи шаблона ...

6
Неэпический
 Аватар для Croessmah
18149 / 10731 / 2067
Регистрация: 27.09.2012
Сообщений: 27,035
Записей в блоге: 1
24.03.2018, 22:06
Лучший ответ Сообщение было отмечено Undisputed как решение

Решение

Цитата Сообщение от Undisputed Посмотреть сообщение
C++
1
const char* cstr = stream.str().c_str();
stream.str() вернет копию строки, которая сдохнет опосля всего выражения. Указатель, естественно протухнет и будет вонять на всю программу.
1
901 / 478 / 93
Регистрация: 10.06.2014
Сообщений: 2,700
24.03.2018, 22:13  [ТС]
Croessmah,
Капец, точно! Получается тут UB, поэтому результаты такие.
Спасибо!
0
309 / 221 / 74
Регистрация: 23.05.2011
Сообщений: 981
24.03.2018, 22:14
Начнём с того, что этот код некорректный:
Цитата Сообщение от Undisputed Посмотреть сообщение
const char* cstr = stream.str().c_str();
http://www.cplusplus.com/refer... tream/str/
http://ru.cppreference.com/w/c... ring/c_str
string str() const;
The first form (1) returns a string object with a copy of the current contents of the stream.

const CharT* c_str() const
Указатель на лежащий в основе строки массив, такой, что data() + i == &operator[](i) для каждого i в [0, size()]. (начиная с C++11)
Поясняю: str() создаёт копию строки, а c_str даёт указатель на массив её символов. Но так как ты никуда не записываешь результат str(), то после выполнения строки 11 память по адресу c_str() очищается. Значит, по указателю cstr у тебя хранится мусор, результат выполнения strlen над которым неопределён.
В 19-ой строке же деструктор строки вызывается уже после вывода, поэтому всё нормально выводит.

Железное правило таково: используй результат вызова c_str только в области видимости объекта std::string и до первой модификации объекта.
1
901 / 478 / 93
Регистрация: 10.06.2014
Сообщений: 2,700
24.03.2018, 22:27  [ТС]
New man,
Да, я понял в чем дело
0
309 / 221 / 74
Регистрация: 23.05.2011
Сообщений: 981
24.03.2018, 22:31
Undisputed, да блин, вы так быстро разговариваете, что успели разобраться до того, как я дописал
0
Неэпический
 Аватар для Croessmah
18149 / 10731 / 2067
Регистрация: 27.09.2012
Сообщений: 27,035
Записей в блоге: 1
24.03.2018, 22:44
Цитата Сообщение от New man Посмотреть сообщение
да блин, вы так быстро разговариваете, что успели разобраться до того, как я дописал
0
Надоела реклама? Зарегистрируйтесь и она исчезнет полностью.
raxper
Эксперт
30234 / 6612 / 1498
Регистрация: 28.12.2010
Сообщений: 21,154
Блог
24.03.2018, 22:44
Помогаю со студенческими работами здесь

Странное поведение операции XOR и std::cin
Объясните, пожалуйста, почему этот код работает правильно: /* Обмен значений двух переменных без использования дополнительной переменной...

Запрошено преобразование от ‘const std::string*’ к нескалярному типу ‘std::string’
private: std::string firstName; }; std::string ClientData::getFirstName() const{ return firstName; } Дает в итоге...

Реализация класса MyString. Стандартная библиотека, std::string, std::vector
как добавить реализацию конкатенации строк через перегрузку оператора &quot;+=&quot; в классе MyString и почему ошибка выдается???#include...

Передача функции указатель на элемент std::vector<std::string>
Доброй ночи тем, кому не спится (или живет в другом часовом поясе:p)! Есть функция, требующая в качестве параметра указатель на...

Операция std::cout для Объекта типа std::string
Кто детально объяснит почему не выводит ? Дает вот так &quot;Отсутствует оператор &quot;&lt;&lt;&quot;, соответствующий этим операндам&quot; ...


Искать еще темы с ответами

Или воспользуйтесь поиском по форуму:
7
Ответ Создать тему
Новые блоги и статьи
Модульный подход на примере F#
DevAlt 06.03.2026
В блоге дяди Боба наткнулся на такое определение: В этой книге («Подход, основанный на вариантах использования») Ивар утверждает, что архитектура программного обеспечения — это структуры,. . .
Управление камерой с помощью скрипта OrbitControls.js на Three.js: Вращение, зум и панорамирование
8Observer8 05.03.2026
Содержание блога Финальная демка в браузере работает на Desktop и мобильных браузерах. Итоговый код: orbit-controls-threejs-js. zip. Сканируйте QR-код на мобильном. Вращайте камеру одним пальцем,. . .
SDL3 для Web (WebAssembly): Синхронизация спрайтов SDL3 и тел Box2D
8Observer8 04.03.2026
Содержание блога Финальная демка в браузере. Итоговый код: finish-sync-physics-sprites-sdl3-c. zip На первой гифке отладочные линии отключены, а на второй включены:. . .
SDL3 для Web (WebAssembly): Идентификация объектов на Box2D v3 - использование userData и событий коллизий
8Observer8 02.03.2026
Содержание блога Финальная демка в браузере. Итоговый код: finish-collision-events-sdl3-c. zip Сканируйте QR-код на мобильном и вы увидите, что появится джойстик для управления главным героем. . . .
Реалии
Hrethgir 01.03.2026
Нет, я не закончил до сих пор симулятор. Эта задача сложнее. Не получилось уйти в плавсостав, но оно и к лучшему, возможно. Точнее получалось - но сварщиком в палубную команду, а это значит, в моём. . .
Ритм жизни
kumehtar 27.02.2026
Иногда приходится жить в ритме, где дел становится всё больше, а вовлечения в происходящее — всё меньше. Плотный график не даёт вниманию закрепиться ни на одном событии. Утро начинается с быстрых,. . .
SDL3 для Web (WebAssembly): Сборка библиотек: SDL3, Box2D, FreeType, SDL3_ttf, SDL3_mixer и SDL3_image из исходников с помощью CMake и Emscripten
8Observer8 27.02.2026
Недавно вышла версия 3. 4. 2 библиотеки SDL3. На странице официальной релиза доступны исходники, готовые DLL (для x86, x64, arm64), а также библиотеки для разработки под Android, MinGW и Visual Studio. . . .
SDL3 для Web (WebAssembly): Реализация движения на Box2D v3 - трение и коллизии с повёрнутыми стенами
8Observer8 20.02.2026
Содержание блога Box2D позволяет легко создать главного героя, который не проходит сквозь стены и перемещается с заданным трением о препятствия, которые можно располагать под углом, как верхнее. . .
КиберФорум - форум программистов, компьютерный форум, программирование
Powered by vBulletin
Copyright ©2000 - 2026, CyberForum.ru