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

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

#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失败)每次我尝试这个。

您的格式字符串要求字符 64 跟在整数之后。 %lu 是格式说明符; 64 是文字字符。

这里有两个问题。第一

%lu64

应该是

"%" SCNu64 

读入 64 位整数。

第二期是

%[B, K, M, G]

需要一个 char*wchar_t* 作为它的输出参数,因为它填充了一个 c 字符串。你需要改变

char c;

至少

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

为了捕获K。我们把所有这些放在一起,我们得到

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

请注意

%[B, K, M, G]

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

%[BKMG]

并得到相同的结果。

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

通用解决方案使用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