c++ - 整数到字符串的转换问题

标签 c++ crypto++

我在使用 Crypto++ 的 Integer 类时遇到了一些问题。我使用的是最新版本 5.6.2。

我正在尝试使用以下代码将整数转换为字符串:

CryptoPP::Integer i("12345678900987654321");

std::ostrstream oss;
oss << i;
std::string s(oss.str());
LOGDEBUG(oss.str()); // Pumps log to console and log file

输出似乎有额外的垃圾数据:

12345678900987654321.ÍÍÍÍÍÍÍÍÍÍÍýýýý««««««««îþîþ

当我直接输出到控制台时,我得到了同样的结果:

std::cout << "Dec: " << i << std::endl; // Same result

此外,我无法使用精度或科学记数法。以下将输出相同的结果:

std::cout.precision(5); // Does nothing with CryptoPP::Integer
std::cout << "Dec: " << std::setprecision(1) << std::dec << i << std::endl;
std::cout << "Sci: " << std::setprecision(5) << std::scientific << i << std::endl;

最重要的是,足够大的数字会破坏整个事情。

CryptoPP::Integer i("12345");

// Calculate i^16
for (int x = 0; x < 16; x++)
{
    i *= i;
}

std::cout  << i << std::endl; // Will never finish

最终,我试图获得一些可以处理大型 Integer 数字的东西,并且可以以科学记数法输出字符串。我对提取 Integer 库或根据需要修改它没有任何问题,但我更喜欢使用稳定的代码。

我是不是做错了什么,或者有什么方法可以让它正常工作吗?

最佳答案

I'm attempting to convert Integer to string with the following code:

CryptoPP::Integer i("12345678900987654321");

std::ostrstream oss;
oss << i;
std::string s(oss.str());
LOGDEBUG(oss.str()); // Pumps log to console and log file

The output appears to have extra garbage data:

12345678900987654321.ÍÍÍÍÍÍÍÍÍÍÍýýýý««««««««îþîþ

我无法在 Visual Studio 2010 上使用 Crypto++ 5.6.2 重现此问题。损坏的输出可能是其他问题的结果,而不是 Crypto++ 中的错误。如果您还没有这样做,我建议尝试仅使用 CryptoPP::Integerstd::cout 在最小程序中重现它,而不是使用其他应用程序代码,以消除所有其他可能的问题。如果它在简单的独立测试中不起作用(这会令人惊讶),则库的构建方式可能存在问题(例如,它可能是使用与您的应用程序所使用的不同的 C++ 运行时或编译器版本构建的) .如果您的独立测试通过,您可以添加其他字符串操作、日志代码等,直到找到罪魁祸首。

我确实注意到您使用的是已弃用的 std::ostrstream 。您可能想要使用 std::ostringstream insteadThis Stack Overflow answer to the question "Why was std::strstream deprecated?" 可能会引起您的兴趣,甚至可能是该答案中提到的问题导致了您的问题。

Additionally, I cannot get precision or scientific notation working. The following will output the same results:

std::cout.precision(5); // Does nothing with CryptoPP::Integer
std::cout << "Dec: " << std::setprecision(1) << std::dec << i << std::endl;
std::cout << "Sci: " << std::setprecision(5) << std::scientific << i << std::endl;

std::setprecisionstd::scientific modify floating-point input/output 。因此,对于 C++ 中的常规整数类型,如 intlong long,这也不起作用(但我可以看到,特别是对于任意长度的整数,如 CryptoPP:Integer 能够以指定精度以科学记数法输出是有意义的)。

即使 C++ 没有这样定义它,Crypto++ 的实现仍然需要注意这些标志。通过查看 std::ostream& operator<<(std::ostream& out, const Integer &a) 的 Crypto++ 实现,我可以看到它识别的唯一 iostream 标志是 std::ios::octstd::ios::hex(分别用于八进制和十六进制格式数字)。

如果您需要科学记数法,您必须自己格式化输出(或使用不同的库)。

On top of all of this, sufficiently large numbers breaks the entire thing.

CryptoPP::Integer i("12345");

// Calculate i^16
for (int x = 0; x < 16; x++)
{
    i *= i;
}

std::cout  << i << std::endl; // Will never finish

这实际上会计算 i^(2^16) = i^65536 ,而不是 i^16 ,因为在每个循环中,您都将 i 与其新的中间值相乘,而不是与其原始值相乘。此代码的实际结果将是 268,140 位长,因此我预计 Crypto++ 只需要很长时间才能生成该输出。

这里是经过调整以产生正确结果的代码:

CryptoPP::Integer i("12345");
CryptoPP::Integer i_to_16(1);

// Calculate i^16
for (int x = 0; x < 16; x++)
{
    i_to_16 *= i;
}

std::cout << i_to_16 << std::endl;

关于c++ - 整数到字符串的转换问题,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29902020/

相关文章:

c++ - 使用 Crypto++ 将文件的 CRC 计算为数值

c++ - 使用 Crypto++ 解密时原始文本末尾的车库冗余字符

c++ - 打印 _mm_cmpeq_epi8 的输出(__m128i 类型)

c++ - Ramer-Douglas-Peucker 路径简化算法

c++ - 如何概括调用 C++ 中的函数列表?

c++ - 分配了 new 的过滤器/StringSink 是否需要删除?

c++ - 加载 key 时 BER 解码错误

c++ - Doxygen - 如何在没有换行符的情况下结束注意力 block

c++ - 如何在 Chromium Embedded Framework (CEF) 中获取 JS 对象?

android - RSA::PublicKey 加载函数段错误 ART(Android 运行时)