整数到字符串的转换问题

Integer to string conversion issues

我在使用 Crypto++ 的 Integer class 时遇到了一些问题。我使用的是最新版本 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 以及您其他应用程序代码的 none 在最小程序中重现它,以消除所有其他可能的问题。如果它在简单的独立测试中不起作用(这会令人惊讶),则库的构建方式可能存在问题(例如,它可能是使用与您的应用程序所使用的不同的 C++ 运行时或编译器版本构建的) .如果您的独立测试通过,您可以添加其他字符串操作、日志代码等,直到找到罪魁祸首。

我确实注意到您使用的 std::ostrstream which is deprecated. You may want to use std::ostringstream instead. This 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;
LOGDEBUG(oss.str()); // Pumps log to console and log file

The output appears to have extra garbage data:

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

我怀疑你所展示的内容与你在现实生活中所做的相比略有简化。我认为问题与 LOGDEBUGostringstream 有关。而且我相信您正在输出 char*,而不是 string(尽管我们还没有看到您的记录器的代码)。

oss.str() 返回的 std::string 是临时的。所以这个:

LOGDEBUG(oss.str());

与此略有不同:

string t(oss.str());
LOGDEBUG(t);

当您打算使用 string 时,您应该始终在 ostringstream 中复制它。或者确保使用包含在一个声明中。

我发现的最好方法是:

// Note: reference, and the char* is used in one statement
void LOGDEBUG(const ostringstream& oss) {
    cout << oss.str().c_str() << endl;
}

// Note: copy of the string below
void LOGDEBUG(string str) {
    cout << str.c_str() << endl;
}

不能甚至这样做(这一点在生产中咬我):

const char* msg = oss.str().c_str();
cout << msg << endl;

您不能这样做,因为从 oss.str() 返回的 string 是临时的。所以 char* 在语句执行后是垃圾。

修复方法如下:

const string t(oss.str());
const char* msg = t.c_str();
cout << msg << endl;

如果你在你的程序中使用 运行 Valgrind,那么你可能会得到与你使用 ostringstreamstrings.

这是一个类似的日志记录问题:stringstream temporary ostream return problem. Also see Turning temporary stringstream to c_str() in single statement. And here was the one I experienced: Memory Error with std:ostringstream and -std=c++11?


正如 Matt 在下面的评论中指出的,您应该使用 ostringstream,而不是 ostrstreamostrstream 自 C++98 以来已被弃用,您应该在使用它时收到警告。

所以改用这个:

#include <sstream>
...

std::ostringstream oss;
...

但我认为问题的根源在于您在 LOGDEBUG 函数或宏中使用 std::string 的方式。


你其他关于Integer的问题在Softwareness的回答和相关评论中已经处理。所以我不会再重复它们了。