1. char c = 128,用 %d 打印,结果是什么? 为什么
十进制:128 二进制:1000 0000
将 128 赋值给 c ,那么1000 0000就为 c 存储的值(电脑存的是补码)
求打印的结果(打印的是源码):
补码:1000 0000 (补码减 1 得到反码,符号位不变)
反码:1111 1111(反码取反得到源码,符号位不变)
源码:1000 0000 (二进制) 128(十进制)
因为是一个带符号的char类型,所以符号位有意义(0为正数 1为负数)
符号位为 1 为负数,所以结果为 -128
用程序验证结果:
#include <stdio.h>
int main(int argc, const char *argv[])
{
char c = 128;
printf("c = %d\n",c);
return 0;
}
2.unsigned char c = -1,用 %u 打印,结果是什么? 为什么?
unsigned char 代表没有符号的 char 类型 (unsigned 类型的都用 %u 打印)
将 -1 转为补码存到 char 内,再打印源码
-1 源码:1000 0001 (反码 = 源码取反)
反码:1111 1110 (补码 = 反码 + 1)
补码:1111 1111 (二进制) 255(十进制)
因为是 unsigned char 类型,所以最高位的符号位无意义
电脑存储的为 255 ,将 255 转为源码输出
255 补码:1111 1111
反码:1111 1111
源码:1111 1111
255 为正数,源码=反码=补码
结果为255,运行程序验证结果:
#include <stdio.h>
int main(int argc, const char *argv[])
{
unsigned char c = -1;
printf("c = %u\n",c);
return 0;
}
3. char 类型数据中存储 0x9C ,用 %d 打印的值是什么? 写出运行结果 (补码,反码,源码推算过程写出来)
0x9c(十六进制) 0x是十六进制的标识,值为9c
9c可以每个数拆分开转化为二进制:1001 1100
电脑存储的为补码,打印的值为源码
补码:1001 1100 (补码 -1 得到反码)
反码:1001 1011 (反码取反得到源码)
源码:1110 0100 (二进制) 100(十进制)
符号位不加入计算,得到十进制为100,
符号位为 1 ,100 为负数,结果为 -100
运行程序验证结果:
#include <stdio.h>
int main(int argc, const char *argv[])
{
char c = 0x9c ;
printf("%d\n",c);
return 0;
}
总结:
1.没有符号位的情况下,最高位加入计算
2.有符号位的情况下,最高位不加入计算(-128和-0除外,最高位既是符号位又是数字)
3.其中正数的 源码 = 反码 = 补码
4.正数的两种情况:
1)没有符号位,全为正数
2)二进制的符号位为 0