Я хотел бы декодировать строку большого файла в C ++.
Размер цепочки составляет: 1827500 знаков
и файл: 1370626 байт
Моя проблема в том, что функция декодирования не работает.
Декодированный файл отличается от исходного файла.
Ниже приведены два способа, которые я пробовал
Первый :
char *base64_decode(const std::string &input, long *size)
{
BIO *p_bio_mem = nullptr;
BIO *p_bio_b64 = nullptr;
char *retrnvalue;
p_bio_b64 = BIO_new(BIO_f_base64());
if (!p_bio_b64) { throw std::runtime_error("BIO_new failed"); }
BIO_set_flags(p_bio_b64, BIO_FLAGS_BASE64_NO_NL); //Don't require trailing newlines
p_bio_mem = BIO_new_mem_buf((void*)input.c_str(), input.length());
if (!p_bio_mem) { throw std::runtime_error("BIO_new failed"); }
BIO_push(p_bio_b64, p_bio_mem);
// read result from chain
// read sequence (reverse to write): buf <<-- p_bio_b64 <<-- p_bio_mem
std::vector<char> buf((input.size() * 3 / 4) + 1);
std::string result;
for (;;)
{
auto nread = BIO_read(p_bio_b64, buf.data(), buf.size());
if (nread < 0) { return NULL; //fail}
if (nread == 0) { break; } // eof
result.append(buf.data(), nread);
}
BIO_free_all(p_bio_b64);
*size = buf.size();
retrnvalue = new char[*size];
memcpy(retrnvalue, buf.data(), *size);
return retrnvalue;
}
Во-вторых:
Код отсюда: Как base64 кодировать (декодировать) в C?
Некоторые концы строк отличаются:
Но не весь файл:
Ты можешь сказать мне, почему? и / или скажите мне другой способ кодирования файла для легкого переноса?
У меня есть это на входе: drive.google.com/file/d/0B1i4Ez8N86wFblJnaFF6YVNVTWs/view
И я хочу это в выходной: drive.google.com/file/d/0B1i4Ez8N86wFdl9OUE5UMFB3R28/view
(извините, я не могу поставить больше 2 ссылок)
PS: когда я декодирую с помощью «certutil -decode» в пакетном режиме, это работает без проблем.
Решено: проблема решена, проблема была написана. Исправлено с функцией записи ofstream
Ваш код для чтения данных в формате base64 довольно запутан:
std::vector<char> buf((input.size() * 3 / 4) + 1);
std::string result;
for (;;)
{
auto nread = BIO_read(p_bio_b64, buf.data(), buf.size());
if (nread < 0) { return NULL; } //fail
if (nread == 0) { break; } // eof
result.append(buf.data(), nread);
}
BIO_free_all(p_bio_b64);
*size = buf.size();
retrnvalue = new char[*size];
memcpy(retrnvalue, buf.data(), *size);
return retrnvalue;
Вы читаете свои данные в строку под названием result
, но вы никогда ничего с этим не делаете. Затем вы копируете содержимое вашего чистого буфера в выходной буфер. Вы также можете покончить с вашим size
указатель, и просто вернуть std::string
вместо сырого char*
,
Другая возможная проблема — линия
BIO_set_flags(p_bio_b64, BIO_FLAGS_BASE64_NO_NL); //Don't require trailing newlines
Если ваш ввод содержит новые строки, то эта строка вызовет проблемы. Если вы хотите работать с вводом, который может содержать или не содержать символы новой строки, вам нужно сделать эту строку условной:
if (input.find('\n') == std::string::npos) {
BIO_set_flags(p_bio_b64, BIO_FLAGS_BASE64_NO_NL); //Don't require trailing newlines
}
Все вместе, это будет выглядеть примерно так:
std::string base64_decode(const std::string &input)
{
BIO *p_bio_mem = nullptr;
BIO *p_bio_b64 = nullptr;
p_bio_b64 = BIO_new(BIO_f_base64());
if (!p_bio_b64) { throw std::runtime_error("BIO_new failed"); }
if (input.find('\n') == std::string::npos) {
BIO_set_flags(p_bio_b64, BIO_FLAGS_BASE64_NO_NL); //Don't require trailing newlines
}
p_bio_mem = BIO_new_mem_buf((void*)input.c_str(), input.length());
if (!p_bio_mem) { throw std::runtime_error("BIO_new failed"); }
BIO_push(p_bio_b64, p_bio_mem);
std::stringstream result;
std::vector<char> buf(1024);
while (auto nread = BIO_read(p_bio_b64, buf.data(), buf.size()))
{
if (nread < 0) { throw std::runtime_error("OMGZ"); } //fail
result.write(buf.data(), nread);
}
BIO_free_all(p_bio_b64);
return result.str();
}
Вы также, вероятно, захотите добавить обработку ошибок, чтобы очистить BIO
случаи в случае ошибки (как в ответе, на который вы ссылались), но это не то, что приводит к неправильному результату.
Решено: проблема решена, проблема была написана. Исправлено с функцией записи ofstream