数据类型
整型家族(char…)
浮点型家族
构造类型 - 自定义类型
- 数组
- struct 结构体类型
- enum 枚举
- union 联合体
指针类型
空类型
数据类型决定了内存开辟空间的大小
强制类型转化
大端和小端
大端:低位放在高地址
小端:低位放在低地址
如何判断大小端
曾经有一道笔试题:
请简述大小端字节序的概念,设计一个小程序来判断当前机器的字节序。
int check_sys()
{
int a = 1;
char* p = (char*)&a;
return *p;//返回首地址的值
}
int main()
{
int ret = check_sys();
if (ret == 1)
{
printf("小端");
}
if (ret == 0)
{
printf("大端");
}
return 0;
}
数据类型的存储大小和值范围
数据类型是有取值范围的。
char在内存中的存储
下面用图片来介绍一下char类型的值范围:
从这张图片就可以清楚地看出char的取值范围没有128。因为128时二进制序列补码为10000000,char默认为有符号的,所以1代表负数,因此不会有128的存在。
练习
int main()
{
char a = -1;
signed char b = -1;
unsigned char c = -1;
printf("%d %d %d", a, b, c);//%d - 需整型提升
}
int main()
{
char a = -128;
printf("%u\n", a);
return 0;
}
整型提升后以%u形式打印,此时高位不视作符号位,故原反补码一样。
int main()
{
int i = -20;
unsigned int j = 10;
printf("%d\n", i + j);
return 0;
}
11111111111111111111111111101100 - i的补码
j为无符号整型,原反补码相同
00000000000000000000000000001010 - j的补码
两补码相加为
11111111111111111111111111110110 - 补码(%d为有符号的十进制数)
补码变为原码为:
10000000000000000000000000001010 – -10
int main()
{
unsigned int i;
for (i = 9; i >= 0; i--)
{
printf("%u\n", i);
}
return 0;
}
unsigned int 为无符号类型,不会出现负数,即始终大于等于0。所以会陷入死循环。其实是取决于输出时打印的类型。
int main()
{
char a[1000];
int i;
for (i = 0; i < 1000; i++)
{
a[i] = -1 - i;
}
printf("%d", strlen(a));
return 0;
}
这道题取决于char的取值范围。char的取值范围看作一个圆(即一个循环)。
char的值只有这些可能,当strlen遇到0时,结束长度计算。
所以128+127 = 255
巧记口诀:
超出范围的数据如果是正数,则减去256;如果是负数。则加上256。
unsigned char i = 0;
int main()
{
for (i = 0; i <= 255; i++)
{
printf("hello\n");
}
return 0;
}
char的最大值为255,即永远小于255,所以会陷入死循环。
float在内存中的存储
练习
int main()
{
int n = 9;
float* pfloat = (float*)&n;
printf("n = %d\n", n);
printf("pfloat = %f\n", *pfloat);
*pfloat = 9.0;
printf("num = %d\n", n);
printf("pfloat = %f\n", *pfloat);
return 0;
}
n刚开始是整型,第一次打印*pfloat时,要按浮点型的存储来看;
第二次将浮点型存入 *pfloat,二进制表示为:
0 10000010 00100000000000000000000 - 也是补码
十进制打印n为:
按浮点数存进去,再打印浮点数自然为9.000000
浮点数以二进制方式表示
IEEE标准规定:
- S:代表正负号,正数时为0,负数时为1
- E:2^E代表指数位
- W:M代表有效数字,1~2
- 存入内存时E的真实值必须加上一个中间数,单精度中间数为127;双精度中间数为1023
例如:浮点数5.5
101.1 - 二进制表示 - 1.011* 2^2
S = 0 | M = 1.011 | E = 2
S = 0 | M = 011 | E = 2+127 - (1忽略)
实际往内存中存储的是
0 100 0000 1 011 0000 0000 0000 0000 0000
十六进制即
40 b0 00 00
上面为存入时的情况,下面介绍取出时的情况:
(黄色二进制部分为E)
E不全为0或不全为1(▲)
E的值减去123(或1023),再在M前加上第一位的1。E全为0
此时E的真实值为-127,E代表指数,即为一个无限接近0的数。所以,此时,E等于1-127
或1-1023
,M不再加上第一位的1。E全为1
此时M全为0,表示±无穷大。
例如:0 10000001 01100000000000000000000
- 符号位为0,表示是正数;
- 指数位为10000001,换算成十进制为129,所以指数为129-127=2;
- 尾数位为01100000000000000000000,换算成十进制为
1(即在M前加上1)+1/4+1/8
; 所以相应的十进制数值为:2^2*(1+1/4+1/8)=4+1+1/2=5.5