这个问题涉及到早期的C语言设计。也就是在第一个C标准(C89)诞生以前的K&R C。
那时候的 C 语言不需要声明函数原型,函数参数在传入时会被转化,具体转化规则有两个:
第一:所有小于 int 类型的整数都会被转化到 int 型 或者 unsigned int。
第二:所有 float 会被转化为 double。
无论你实际上输入的是 char,还是short,函数调用的时候,这些参数都会被转化为 int 并传入。
同样,无论你输入的 是 float 还是 double,函数调用时实际都以 double 传入。
以上规则作用于两类函数:第一是没有函数原型,或者函数原型内没有参数类型的函数。第二是变参函数中的可变参数。
在现代C语言中,C89以后开始标准建议是不会使用无函数原型的函数,但变参函数依然存在,所以这种自动转化输入参数的调用规则实际上在 printf 这样的函数中被保留了下来。
printf 是一个变参函数,其中的可变参数也符合转化原则,因此实际上所有小于 int 的类型最终都会被转化为 int,小于 double 的浮点都会被转化为 float 。
所以当处理 format 的时候,后面可变参数的传入的值并不是你传入的实际类型。你必须在 format 中指定要显示为何等参数类型,才能正确显示。