1 十进制
#include <stdio.h>
int main()
{
char ch1 = 129;
printf("%d\n", ch1);
}
输出
-127
在第4行char ch1 = 129;
中,由于char默认有符号,因此ch1能保存的数据范围是[-128, 127],很明显,129已经越界。由于129是十进制数,默认其为int型,其原码为b0000 0000 0000 0000 0000 0000 1000 0001,随后将最低的8位存入ch1对应的内存,因此ch1保存的是b1000 0001;打印时,系统会将b1000 0001当成是某个8位有符号整数的二进制补码,将其翻译成原码为b1111 1111,因此是-127。
2 十六进制
#include <stdio.h>
int main()
{
char ch2 = 0x82;
printf("%d\n", ch2);
}
输出
-126
由于0x82是十六进制数,系统会将其原码赋值给ch2,它的原码为b1000 0010;打印时,系统会将b1000 0010当成是某个8位有符号整数的二进制补码,将其翻译成“原码”为b1111 1110,因此是-126。
3 八进制
#include <stdio.h>
int main()
{
char ch3 = 071;
printf("%d\n", ch3);
printf("%x\n", ch3);
}
输出
57
39
八进制的处理方式与十六进制类似,赋值时存入的是原码,071的原码为b111 001,转换成8位时,为b0011 1001,存入内存的就是这个b0011 1001。按十进制打印时,它是十进制57对应的补码,所以打印的是57;按十六进制打印时,它是十六进制数39对应的原码,所以打印的是39.
4 8位转32位
(1)有符号数
#include <stdio.h>
int main()
{
char num = 0x81; //变量num中存的是:1000 0001
printf("%d\n", num); // 按有符号整数打印为-127
printf("%u\n", num); // 按无符号整数打印为4294967169
return 0;
}
因为%d对应的类型是int型,int有32位,但num实际只有8位,因此printf("%d\n", num)
时,先将num转成32位,再打印,转换过程如下:8位补码(1000 0001) -> 8位原码(1111 1111) -> 32位原码 -> 32位补码
32位原码为:1000 0000 0000 0000 0000 0000 0111 1111 (将符号位移动到最前面,中间用0填充)
32位补码为:1111 1111 1111 1111 1111 1111 0000 0001
按%u打印时,一律按照整数打印,因此为4294967169
(2)无符号数
unsigned char e = 0xfe;
printf("%u\n", e); //结果为多少? 254
printf("%d\n", e); //用%d打印无符号数 254
%u和%d打印的都是32位,但e只有8位,打印时先将其转化为32位,然后再打印
无符号数的转化:8位(1111 1110)-> 32位(0000 0000 0000 0000 0000 0000 1111 1110)转换的过程中无需考虑符号位,直接在前面补0
%u和%d打印,只是对变量符号位的处理方式不一样,不影响8位转32位的过程。
5 结论
(1)C语言中对变量进行赋值,如果赋的是十进制数,则将其补码存入变量,如果赋的是十六进制数或八进制数,则将其原码存入变量;
(2)C语言使用%d进行十进制打印时,则一律将内存中的数按“补码——原码——十进制”的方式进行处理;
(3)使用%x或者%o打印时,则一律将内存中的数按“原码——十六进制或八进制”的方式进行处理。