Стек вокруг переменной "temp" поврежден

Я пытаюсь преобразовать std::string, содержащий несколько значений, разделенных пробелами, в векторный объект. У меня это работает (имеется в виду, что отладчик показывает правильные значения в векторе в конце алгоритма), однако я получаю исключение, которого я никогда раньше не видел. Исключение ниже:

Исключение я получаю

Я никогда не видел этого раньше, и я целый час изучал причины этого, и до сих пор не понимаю этого.

Мой алгоритм, который вызывает исключение, выглядит следующим образом:

vector<BYTE> ConvertStringOfValuesToVectorOfBytes(std::string valStr)
{
    vector<string> vectorOfValues = split(valStr, ' '); //split string into vector of strings (works)
    vector<BYTE> Hex;
    vector<const char*> vectOfCStrings;

    for(int i = 0; i < vectorOfValues.size(); i++)
    {
        const char* temp = vectorOfValues[i].c_str();
        vectOfCStrings.push_back(temp);
    }

    //now we have a vector of c strings...

    for(int i = 0; i < vectOfCStrings.size(); i++)
    {       
        char temp = 0;
        sscanf(vectOfCStrings[i], "%x", &temp);
        Hex.push_back(temp);
    }

    return Hex;

} //<-- debugger gets to here and on the next step causes the exception

Я понятия не имею, как я могу решить эту проблему. Как я уже сказал, алгоритм "работает" в том смысле, что я получаю правильный результат, мне просто нужно прояснить исключение. Как это исправить?

4 ответа

Эта строка не верна:

sscanf(vectOfCStrings[i], "%x", &temp);

Формат %x должен быть предназначен для использования для intне char,

Вы получаете повреждение стека, потому что sscanf ожидает адрес int, Он пытается использовать адрес, который вы передаете как способный удерживать int, Очевидно, он использует больше памяти, чем допустимо.

Что вам нужно это:

for(int i = 0; i < vectOfCStrings.size(); i++)
{       
    int temp = 0;
    sscanf(vectOfCStrings[i], "%x", &temp);
    Hex.push_back(static_cast<char>(temp));
}

Culprit, кажется, в этих двух строках:

    char temp = 0;
    sscanf(vectOfCStrings[i], "%x", &temp);

Именно поэтому семейство функций printf/scanf не используется в C++ и заменено на std::iostream механизм. функция sscanf ожидает получить указатель на unsigned int, когда вы укажете %x но вы передаете указатель на символ, и нет никакого способа sscanf Сам может подтвердить это. Хотя некоторые современные компиляторы могут проверять, переданы ли значения в printf или же scanf сопоставляет спецификаторы формата и генерирует предупреждения в противном случае, что в общем случае не работает (например, формат передается функции вместо использования строкового литерала). Так вот sscanf указатель разыменования &temp как указатель на unsigned int и перезаписывает память вокруг временной переменной как sizeof char == 1.

Что именно ваш алгоритм пытается достичь?

Судя по всему, вы разбиваете std:: string на std::vector из 'words', а затем конвертируете эти слова в строки в стиле c (почему?), Затем собираете std:vector, содержащий первую букву каждого слова, интерпретируемого как int. Вместо этого используйте std: stringstream:

for ( auto& word : vectorOfValues )
{
    for ( auto& letter : word ) 
    {
        unsigned int val;
        std::stringstream ss;
        ss << std::hex << word[0];
        ss >> val;
        Hex.push_back( static_cast<BYTE>( val ) );
    }
}

Как правило, если вы не взаимодействуете напрямую с C API, вам не следует использовать строки в стиле c в C++.

std::vector<unsigned char> func(std::string valStr) {
    vector<string> vectorOfValues = split(valStr); //split string into vector of strings (works)
    vector<BYTE> Hex;
    vector<const char*> vectOfCStrings;

    for(int i = 0; i < vectorOfValues.size(); i++) {
        const char* temp = vectorOfValues[i].c_str();
        vectOfCStrings.push_back(temp);
    }

    //now we have a vector of c strings...

    for(int i = 0; i < vectOfCStrings.size(); i++) {
        int temp = 0;
        sscanf(vectOfCStrings[i], "%x", &temp);
        Hex.push_back(char(temp));
    }

    return Hex;
}

Так что да, я только что сделал второй tmp Int, потому что sscanf и тому подобное int как типы, а так как byte меньше, чем int, что вызывает переполнение.

Не уверен, что это работает, но это больше не дает мне ошибки.

Другие вопросы по тегам