Linux下C语言的一道经典面试题

如果在Linux下使用GCC编译器执行下列程序,输出结果是什么?

#include<stdio.h>
int main(){
    char c=127;
    printf("%d",++c);
    printf("%d",++c);
    return 0;
}

只是知道涉及到类型转换、数据截断填充。但不知道具体解释?

原问题出处:Linux下C语言的几道经典面试题

阅读 5.6k
3 个回答

char 的长度是 1 字节,并且,大多数的机器都将其视为有符号数的,所以它的表示范围是 [-128, 127](参见《深入理解计算机系统》P27~P28)。所以,当你给 c 赋值 127 时,你执行了 ++c,那么就会导致溢出,因为它只有一个字节。

在机器中表示 127,变成二进制是这样的 01111111,可以看到,当你加 1 之后,结果就变成了 10000000,由于在计算机内部,负数是用补码来表示的,所以就变成了 -128。之后再 ++c,就是 -127 了。

至于不同的类型,其实在计算机内部表现的都是一样的,都是一块内存。所以类型不是限制。

这题考的是 编译器,而不是语言。
定义3个变量:

char c; // 表示一个 ascii字符
signed char s_c;   //有符号 1字节 整数
unsigned char u_c; //无符号 1字节 整数

C语言里,c 在参与计算时, c 转为 s_c 还是 u_c ,是由编译器来决定的。

gcc 认为 c 是 有符号的,后续的计算 和 输出 都是按有符号数进行处理。

建议实际工程中使用平台提供的<stdint.h>里的类型定义,包括uint8_t, int8_t, uint16_t, int16_t, uint32_t, int32_t, uint64_t, int64_t,这样既不会出现char的符号问题,也不会出现long的位数问题。

此外,还需要注意的地方是printf是接受的可变参数,可变参数在传递时会应用"加宽"规则: uint8_t,uint16_t,uint32_t转换为uint32_tint8_t,int16_t,int32_t转换为int32_tfloat,double转换为doubleprintf在内部处理的时候,并不知道实际传过来的数据类型,只能通过format字符串内的%x来判断数据的类型。因此,如果实际需要打印无符号数时,应该使用%ud,而不应该使用%d

撰写回答
你尚未登录,登录后可以
  • 和开发者交流问题的细节
  • 关注并接收问题和回答的更新提醒
  • 参与内容的编辑和改进,让解决方法与时俱进
推荐问题