Чтобы реализовать клиент для некоторого протокола, который ожидает заполнения нулевыми байтами определенной строки, я реализовал функции, которые дополняют строку определенным количеством нулевых байтов:
string padToFill(string s, int fill){
int len = s.length();
if (len < fill){
for (int i = 0; i < fill - len; i++){
s.append("\0");
}
}
return s;
}
Однако на практике результат, который я получаю от этой функции, идентичен строке, которую я получаю в качестве аргумента. Так, например, padToFill("foo", 5) это foo, а не foo\0\0. Ожидается, что даже строки C++ будут обрабатывать нулевые байты, как и любой другой символ.
Как мне добиться желаемого нулевого заполнения?
s.push_back('\0'); Обратите внимание, что std::string уже управляет \0-ограничителем в конце строки, который не является частью длины. Если вы отодвинете дополнительные \0-терминаторы, они будут частью длины (и все еще будет дополнительная неучтенная \0-концевая камбуз).
@ring0 Что касается меня, то я ничего не понял.
извините, я поторопился, сказав, что код в порядке. Это не так. В любом случае, добавление того, как вы узнаете, что функция не работает, было бы хорошо
Вы вызываете эту функцию: string& append (const char* s); поэтому вы добавляете строку c...
@ 463035818_is_not_a_number Я также думал, что отладчик не показал мне нулевые байты. Это не вариант. Я использую gdb с CLion, и я проверил размер строки, и он также не изменился. Я также попытался преобразовать строку в вектор символов, и там я явно вижу, что нулевых байтов нет.
@ring0 В s.append("\0");"\0" обрабатывается так же, как "", поэтому s не меняется. Вместо этого вы можете использовать s.append("\0", 1);, или s.append(1, '\0');, или s.append({'\0'});, но s.push_back('\0'); будет понятнее, как предложил Элджей.





Используемая вами перегрузка append принимает строку C, которая завершается нулем (то есть первый нулевой байт отмечает ее конец).
Самый простой способ здесь, вероятно, использовать другую перегрузку append:
s.append(fill - len, '\0');
Обратите внимание, что для этого требуется C++20. Для более старых стандартов С++ есть
s.resize(fill, '\0');
или просто
s.resize(fill);
Подумайте об этом, в любом случае, это может быть более четкий способ описать то, что вы делаете. Мне нравится второй вариант здесь для ясности.
как вы проверяете строку? У
std::stringможет быть много\0в конце, но есть некоторые подводные камни, которые могут заставить вас поверить, что их нет. Код, который вы разместили, выглядит нормально. Какой код вы использовали, чтобы увидеть, сколько нулевых терминаторов в результате?