一道C语言坑爹题

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

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

有了这个知识点,我们再来看这道题:
char取值范围是:-128~127
一道C语言坑爹题
unsigned char取值范围是:0~255
一道C语言坑爹题
因此,变量c被赋值128,而它在内存中的实际值是-128

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

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

第三次,uc被强制转换为char所以它与c的补全后存储格式都是
一道C语言坑爹题
相加后的结果为
一道C语言坑爹题
所以,第三次输出结果为“0xff00”(因为unsigned short占两个字节,所以只截取最低位两个字节输出

一道C语言坑爹题

上一篇:基础练习 Sine之舞


下一篇:Docker基础之二: Linux快速入门