一道C语言坑爹题

来源:互联网 发布:淘宝店铺联盟报名入口 编辑:程序博客网 时间:2024/05/22 02:29

请写出以下代码输出结果:

#include <stdio.h>int main(){    char c;    unsigned char uc;    unsigned short us;    c = 128;    uc = 128;    us = c + uc;    printf("0x%x\n", us);    us = (unsigned char)c + uc;    printf("0x%x\n", us);    us = c + (char)uc;    printf("0x%x\n", us);    return 0;}

看答案之前,自己试一试

怎么样?有没有被类型转换搞得一脸懵逼?

好吧, 我们来看看到底怎么解:

首先来看看C语言中的类型转换原则:
这里写图片描述
同时在C语言中存在一种Integer Promotion(整型提升)机制:
简单地说就是如果在char, short这类字长短的类型在做运算时会先转换为int型,运算结束后再根据需要的结果类型做截取(具体请查阅详细资料)
这样做的目的是因为在通用CPU的运算器中操作数都是int类型,很难直接实现两个8比特字节直接相加运算,因此需要先转换为int便于CPU做运算

有了这个知识点,我们再来看这道题:
char取值范围是:-128~127
这里写图片描述
unsigned char取值范围是:0~255
这里写图片描述
因此,变量c被赋值128,而它在内存中的实际值是-128

当它被整型提升时,是这样存储的(以下所有超出的字节中的位数都是运算时根据数据类型的自动补全,只在运算时有作用并不参与实际存储):
这里写图片描述
而uc是这样存储的:
这里写图片描述
所以,c+uc为:
这里写图片描述
所以,第一次输出为“0x0”;

第二次,c被强制转换为unsigned char所以它与uc的补全后存储格式都是
这里写图片描述
相加后的结果为
这里写图片描述
所以,第二次输出结果为“0x100”(注意这里是16进制输出,四个二进制位是一个十六进制为,unsigned short占两个字节,也就是占4个十六进制位,每次输出也就是只截取最低位的两个字节

第三次,uc被强制转换为char所以它与c的补全后存储格式都是
这里写图片描述
相加后的结果为
这里写图片描述
所以,第三次输出结果为“0xff00”(因为unsigned short占两个字节,所以只截取最低位两个字节输出

这里写图片描述