为什么 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_t
和 char
,但它只读取了 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
#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_t
和 char
,但它只读取了 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