我正在处理包含大量数字(磁盘偏移)的文件,并且在将scanf
用作%lli
时遇到了问题-数字的格式为0x
...但是scanf
无法读取所有位。
这里是示例代码(我在https://www.onlinegdb.com/online_c_compiler上进行了测试)。
#include <stdio.h>
#include <stdlib.h>
int main()
{
char* str = malloc(30);
unsigned long long l;
sprintf(str, "0x%llx\n", 0xffffffffffffffffULL);
printf("%s\n", str);
sscanf(str, "%lli", &l);
printf("%llx\n", l);
return 0;
}
它输出:
0xffffffffffffffff
7fffffffffffffff
而我希望两个数字相等。
如果使用%llx
说明符,它可以正常工作,但我不想将自己限制为十六进制格式的数字。
不幸的是,i
代表signed整数,并且相应的预期参数是指向int
的指针;使用ll
的宽度将为long long
,但仍将被签名。对于具有基本检测的无符号整数,没有scanf
格式字符。
没有完美的解决方案。您可以使用strtoull(str, &endptr, 0)
(甚至strtoumax
),它可以正确检测格式,但是使用起来比较麻烦。
但是scanf
也不完美-如果数字太大,则总是会得到最大值,就像得到LLONG_MAX
一样,没有错误指示。
[%lli
代表long long signed int
(在您的系统上仅上升到0x7fffffffffffffffff)。
[%llu
用于long long unsigned int
(这是您要读取的内容)。
将前者替换为后者。