如果在Linux下使用GCC编译器执行下列程序,输出结果是什么?
#include<stdio.h>
int main(){
char c=127;
printf("%d",++c);
printf("%d",++c);
return 0;
}
只是知道涉及到类型转换、数据截断填充。但不知道具体解释?
原问题出处:Linux下C语言的几道经典面试题
如果在Linux下使用GCC编译器执行下列程序,输出结果是什么?
#include<stdio.h>
int main(){
char c=127;
printf("%d",++c);
printf("%d",++c);
return 0;
}
只是知道涉及到类型转换、数据截断填充。但不知道具体解释?
原问题出处:Linux下C语言的几道经典面试题
这题考的是 编译器
,而不是语言。
定义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_t
,int8_t,int16_t,int32_t
转换为int32_t
,float,double
转换为double
。printf
在内部处理的时候,并不知道实际传过来的数据类型,只能通过format字符串内的%x来判断数据的类型。因此,如果实际需要打印无符号数时,应该使用%ud
,而不应该使用%d
。
7 回答5.3k 阅读
4 回答4k 阅读
2 回答5.9k 阅读✓ 已解决
2 回答2.6k 阅读✓ 已解决
1 回答2.3k 阅读✓ 已解决
2 回答802 阅读✓ 已解决
1 回答3.3k 阅读
char
的长度是 1 字节,并且,大多数的机器都将其视为有符号数的,所以它的表示范围是[-128, 127]
(参见《深入理解计算机系统》P27~P28)。所以,当你给c
赋值 127 时,你执行了++c
,那么就会导致溢出,因为它只有一个字节。在机器中表示 127,变成二进制是这样的
01111111
,可以看到,当你加 1 之后,结果就变成了10000000
,由于在计算机内部,负数是用补码来表示的,所以就变成了 -128。之后再++c
,就是 -127 了。至于不同的类型,其实在计算机内部表现的都是一样的,都是一块内存。所以类型不是限制。