浮点类型(float、double)、char类型在内存中存储方式

数据类型

整型家族(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-1271-1023,M不再加上第一位的1。
  • E全为1
    此时M全为0,表示±无穷大。

例如:0 10000001 01100000000000000000000

  1. 符号位为0,表示是正数;
  2. 指数位为10000001,换算成十进制为129,所以指数为129-127=2;
  3. 尾数位为01100000000000000000000,换算成十进制为 1(即在M前加上1)+1/4+1/8; 所以相应的十进制数值为:2^2*(1+1/4+1/8)=4+1+1/2=5.5
  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值