为什么 sscanf 无法从一个字符串中读取uint64_t和字符?

Why sscanf can't read an uint64_t and a char from one string?

本文关键字:uint64 读取 字符 字符串 一个 sscanf 为什么      更新时间:2023-10-16
#include <cstdio>
#include <cstdint>
#include <cassert>
int main() {
std::uint64_t ui;
char c;
auto ret = std::sscanf("111K", "%lu64%[B, K, M, G]", &ui, &c);
assert(ret == 2);
assert(ui == 111);
}

我尝试使用sscanf从一个字符串中读取uint64_tchar,但每次尝试时它只读取ui(断言ret == 2失败(。

这里有两个问题。第一个

%lu64

应该是

"%" SCNu64 

以读取64位整数。

第二个问题是

%[B, K, M, G]

在填充c字符串时需要char*wchar_t*作为其输出参数。您需要更改

char c;

至少

char c[2] // 2 because it adds a null terminator

以便捕获CCD_ 8。我们把这些放在一起,我们得到了

auto ret = std::sscanf("111K", "%" SCNu64 "%[B, K, M, G]", &ui, c);

请注意

%[B, K, M, G]

实际上是试图匹配括号内的所有空格和逗号。你也可以把它写成

%[BKMG]

并得到相同的结果。

您的格式字符串需要字符64跟在整数后面。%lu是格式说明符;CCD_ 11是文字字符。

您的格式字符串无效。在64位构建中应该是这样的:当为32位构建时,"%lu%1[BKMG]""%llu%1[BKMG]"

通用解决方案使用SCNu64宏:"%" SCNu64 "%1[BKMG]"。该宏在cinttypes头文件中定义。

此外,您的最后一个参数传递错误。它必须是一个字符数组,因为零将附加在末尾。您的代码会导致未定义的行为,因为当写入c值时,也会写入该变量以外的内容。

#include <cstdio>
#include <cstdint>
#include <cassert>
#include <iostream>
int main() {
std::uint64_t ui;
char s[32];
auto ret = std::sscanf("111K", "%lu%1[BKMG]", &ui, s);
std::cout << "ret=" << ret << "  ui=" << ui << " s=" << s << "n";
return 0;
}

https://wandbox.org/permlink/17vZ8OkydJ7zQmP4
https://wandbox.org/permlink/z21Rbsu4mAseZyS4