计算机中的有符号数有三种表示方法,即原码、反码和补码。
三种表示方法均有符号位和数值位两部分,符号位都是用0表示“正”,用1表示“负”,而数值位三种表示方法各不相同。
原码:
直接将二进制数按照正负数的形式翻译成二进制就可以。
反码:
将原码的符号位不变,其他位依次按位取反即可。
补码:
反码+1得到。
正数的原码、反码和补码都相同。
对于整型来说:数据存放内存中其实存放的是补码。
存储模式分为大端模式和小端模式
大端模式:数据的低位保存在内存的高地址中,而数据的高位保存在内存的低地址中。
小端模式:数据的低位保存在内存的低地址中,而数据的高位保存在内存的高地址中。
练习题
下面程序输出什么?
1、
#include<stdio.h>
int main()
{
char a = -1;
signed char b = -1;
unsigned char c = -1;
printf("a=%d,b=%d,c=%d\n", a, b, c);
return 0;
}
a=-1,b=-1,c=255
a,b,c在内存中存储位1111 1111
但转换位int型时
a,b有符号位,前面补1。为1111 1111 1111 1111 1111 1111 1111 1111
转为原码为1000 0000 0000 0000 0000 0000 0000 0001
输出-1
c为无符号位,前面补0。为0000 0000 0000 0000 0000 0000 1111 1111
输出255
2、
#include<stdio.h>
int main()
{
char a = -128;
printf("%u\n", a);
return 0;
}
4294967168
a在内存中存储为1000 0000
转换为无符号整型,前面补1。为1111 1111 1111 1111 1111 1111 1000 0000
3、
#include<stdio.h>
int main()
{
char a = 128;
printf("%u\n", a);
return 0;
}
4294967168
char类型数据范围为-128~127
故a=-128,其余与第二题同理。
4、
#include<stdio.h>
int main()
{
int i = -20;
unsigned int j = 10;
printf("%d\n", i + j);
return 0;
}
-10
i的补码:1111 1111 1111 1111 1111 1111 1110 1100
j的补码:0000 0000 0000 0000 0000 0000 0000 1010
i+j得:1111 1111 1111 1111 1111 1111 1111 0110
转为原码:1000 0000 0000 0000 0000 0000 0000 1010
5、
#include<stdio.h>
int main()
{
unsigned int i;
for (i = 9; i >= 0; i--)
{
printf("%u\n", i);
}
return 0;
}
死循环
unsigned int的范围为0~4294967295,总大于0。
6、
#include<stdio.h>
#include<string.h>
int main()
{
char a[1000];
int i;
for (i = 0; i < 1000; i++)
{
a[i] = -1 - i;
}
printf("%d\n", strlen(a));
return 0;
}
255
strlen到‘\0’停止
char的范围为-128~127
for循环赋值,-1,-2,.......,-128,127,126,......,1,0
-1~-128长度为128,1~127长度为127
128+127=255
7、
#include<stdio.h>
unsigned char i = 0;
int main()
{
for (i = 0; i <= 255; i++)
{
printf("hello world\n");
}
return 0;
}
死循环
unsigned char的范围为0~255。总小于255