我正在修复我继承的遗留项目中的编译器警告。新编译器是 gcc 版本 4.8.5 20150623 (Red Hat 4.8.5-4) (GCC)。
他们有很多代码如下:
#include <cstdio>
#include <cstring>
struct foobar
{
char field1[10];
char field2[5];
};
int main()
{
struct foobar foo;
memset(&foo, ' ', sizeof(foo));
strncpy(foo.field1, "1234567890", sizeof(foo.field1));
// Produces warning
printf("[%.*s]", sizeof(foo.field1), foo.field1);
return 0;
}
这会产生警告消息:
1_test.c: In function ‘int main()’:
1_test.c:16:49: warning: field precision specifier ‘.*’ expects argument of type ‘int’, but argument 2 has type ‘long unsigned int’ [-Wformat=]
printf("[%.*s]", sizeof(foo.field1), foo.field1);
这对我来说似乎是错误的,因为 '.*' 应该期待 size_t 但显然它没有......
除了必须执行以下操作之外,还有没有办法在全局范围内解决此问题:
// Fixes
printf("[%.10s]", foo.field1);
// Fixes
printf("[%.*s]", static_cast<int>(sizeof(foo.field1)), foo.field1);
最佳答案
正确的解决办法是:
std::cout << std::string( foo.field1, sizeof( foo.field1 ) );
这将输出您想要的内容,并且不会生成任何警告。但更好的解决方案当然是在
std::string
中使用 struct foobar
关于C++ printf 字段宽度说明符 ‘.*’ 期望 int 而不是 size_t,我们在Stack Overflow上找到一个类似的问题:https://stackoverflow.com/questions/37753132/