1、计算机中的符号位
一个数据在计算机中存储是以01方式存储的,怎么表示有符号与无符号呢,很简单,数据类型的最高位用于标识数据的符号
- 最高位为1,表明这个数为负数
- 最高位为0,表明这个数为正数
//2-1.c
#include <stdio.h>
int main() {
char i = -5;
short j = 5;
int k = -7;
printf("%d\n", (i & 0x80) != 0); //最高位为1
printf("%d\n", (j & 0x8000) != 0); //最高位为0
printf("%d\n", (k & 0x80000000) != 0); //最高位为1
return 0;
}
2、有符号数的表示
在计算机内部用补码表示有符号数
- 正数的补码为正数本身
- 负数的补码为负数的绝对值各位取反后加1
举例说明:
8位整数5的补码为:0000 0101
8位整数-7的补码为:1111 1001
16位整数20的补码为:0000 0000 0001 0100
16位整数-13的补码为:1111 1111 1111 0011
3、无符号数的表示
在计算机内部用原码表示无符号数
- 无符号数默认为正数
- 无符号数没有符号位
对于固定长度的无符号数
- MAX_VALUE + 1 → MIN_VALUE
- MIN_VALUE - 1 → MAX_VALUE
4、signed 和 unsigned
signed 表示有符号数,unsigned 表示无符号数,C语言中默认是有符号数,当表示无符号数时,需要用 unsigned 修饰
!!注意:C语言中只有整数类型能够声明为 unsigned 变量。
#include<stdio.h>
int main(){
int i; //默认为带符号整数
signed int j; //显示声明变量为带符号整形
unsigned int k; //声明变量为无符号整形
return 0;
}
5、实验
1、有符号数遇见无符号数
先看一个例子
//2-2.c
#include<stdio.h>
int main(){
unsigned int i = 5;
int j = -10;
if (i + j > 0){
printf("i + j > 0\n");
}
else {
printf("i + j < 0\n");
}
return 0;
}
在这里,有符号数遇见无符号数,变为无符号数了。
结论:当无符号数与有符号数混合计算时,会将有符号数转换为无符号数后再进行计算,结果为无符号数
2、错误的使用unsigned
直接看代码
//2-3.c
#include<stdio.h>
int main(){
unsigned int i = 0;
for (i = 9; i >= 0; i--){
printf("i = %u\n", i);
}
return 0;
}
看似应该打印 9 到 0 的所有数字,实际运行,发现是死循环,循环结束的条件是 i < 0,由于 i 是无符号数,永远不可能小于 0。
6、小结
- 1、有符号数用补码表示
- 2、无符号数用原码表示
- 3、unsigned 只能修饰整数类型的变量
- 4、无符号数和有符号数混合计算,有符号数变为无符号数