昨天同事问了我个 C 语言的问题,没能一下答对。问题可以精简如下:
unsigned char a, b;
a = 0x0f;
b = (~a) >> 4;
printf("0x%x", b);
输出的结果为多少。
当时认为是 0x0f,可实测的结果是 0xff。查了下 C 语言的标准,确实是我错了。
K&R C中关于整型提升(integral promotion)的说明如下:
"A character, a short integer, or an integer bit-field, all either signed or not, or an object of enumeration type, may be used in an expression wherever an integer may be used. If an int can represent all the values of the original type, then the value is converted to int; otherwise the value is converted to unsigned int. This process is called integral promotion."
简单的说就是 char 型、short 型在做算术运算时会被提升为整型或无符号整型。因此 (~a) 被提升为 int 型的取反运算。
对于 MinGW gcc,int 型是 32 位, 因此 (~a) = 0xff ff ff f0, 所以 b = 0xff。
说实话,C 标准在这个问题上有点让人不舒服,不过之所以这样规定也是个历史遗留问题了。虽然不完美,但是我们也只能接受。