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

时间:2018-11-02 13:41:33

标签: c++ char cstdio cstdint

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

3 个答案:

答案 0 :(得分:3)

您的格式字符串期望字符64跟随整数。 %lu是格式说明符; 64是文字字符。

答案 1 :(得分:3)

您在这里有两个问题。首先

%lu64

应该是

"%" SCNu64 

读取64位整数。

第二个问题

%[B, K, M, G]

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

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]

并获得相同的结果。

答案 2 :(得分:3)

您的格式字符串无效。 在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