C的整型算术运算总是至少以缺省整型类型的精度来进行的。为了获得这个精度,表达式中的字符和短整型操作数在使用之前被转换为普通整型,这种转换称为“整型提升”。
首先,我们需要知道整型提升的规则:
1.若是有符号数,则前面8*3位补符号位。
2.若是无符号数,则前面面8*3位补0。
char c = 128; //11111111 11111111 11111111 10000000
printf("%d\n", c); //10000000 00000000 00000000 10000000 (-128)
int main()
{
unsigned char a = 200;//00000000 00000000 00000000 11001000
unsigned char b = 100;//00000000 00000000 00000000 01100100
unsigned char c = 0;
c = a + b; //1 00101100
//00000000 00000000 00000000 00101100 44
printf("%d %d", a + b, c);//00000000 00000000 00000001 00101100 300
system("pause:");
return 0;
}
接下来我们分析一下这道题:
a的二进制位为: 11001000
b的二进制位为: 01100100
a+b二进制位为:1 00101100 (前边的1丢掉)
以%d的形式输出,需要进行一次整型提升,补0,00000000 00000000 00000000 00101100(44)
a整型提升为:00000000 00000000 00000000 11001000
b整型提升为:00000000 00000000 00000000 01100100
a+b:00000000 00000000 00000001 00101100 (300)
一个char类型数字在内存中的存储方式如图所示:
int main()
{
char c;
unsigned char uc;
unsigned short us;
c = 128; //10000000 11111111 11111111 11111111 10000000
uc = 128;//10000000 00000000 00000000 00000000 10000000
us = c + uc;// 00000000 00000000 00000000 00000000 0
printf("0x%x ", us);
us = (unsigned char)c + uc;//00000000 00000000 00000000 10000000
//00000000 00000000 00000000 10000000
//00000000 00000000 00000001 00000000(0 1 0 0)
printf("0x%x ", us);
us = c + (char)uc; //11111111 11111111 11111111 10000000
//11111111 11111111 11111111 10000000
//11111111 11111111 11111111 00000000(f f 0 0)
printf("0x%x ", us);
system("pause");
return 0;
}
分析一下:
c的二进制: 10000000(有符号,补1)11111111 11111111 11111111 10000000
uc的二进制:10000000(无符号,补0) 00000000 00000000 00000000 10000000
us = c + uc; us的二进制: 00000000 00000000 00000000 00000000(0 0 0 0)
c的二进制: 10000000(无符号,补0) 00000000 00000000 00000000 10000000
uc的二进制:10000000(无符号,补0) 00000000 00000000 00000000 10000000
us = (unsigned char)c + uc; us的二进制:00000000 00000000 00000001 00000000 (0 1 0 0)
c的二进制: 10000000(有符号,补1)11111111 11111111 11111111 10000000
uc的二进制: 10000000(有符号,补1)11111111 11111111 11111111 10000000
us = c + (char)uc; us的二进制:11111111 11111111 11111111 00000000(f f 0 0)
综上所述,输出结果为:0x0 0x100 0xff00