一道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语言中存在一种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占两个字节,所以只截取最低位两个字节输出

这里写图片描述

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值