C—数据的储存(下)

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6

文章目录


前言

个人主页@小沈熬夜秃头中୧⍤⃝❅
小编介绍欢迎来到我的乱七八糟小星球
专栏C语言学习
本章内容C-数据的储存下
送给各位学无止境
记得 评论 +点赞 +收藏 +关注哦~


一、练习一下

1.例一

看过C语言学习第十五弹后我们一起来做一下下面这道例一

#include<stdio.h>
#include<windows.h>
int main()
{
	unsigned int i;
	for (i = 9; i >= 0; i--)
	{
		printf("%u\n", i);
		Sleep(1000);//单位毫秒
	}
	return 0;
}

答案死循环
解析因为这是一个无符号int 类型的i所以当i=0再- -时成为-1但是无符号整形中没有符号位所以就变成了一个很大的数一直打印这个条件恒成立所以就死循环了
在这里插入图片描述

2.例二

看过例一我们来做一下例二

#include<stdio.h>
#include<windows.h>
int main()
{
	unsigned int i;
	for (i = 9; i >= 0; i--)
	{
		printf("%d\n", i);
		Sleep(1000);//单位毫秒
	}
	return 0;
}

答案死循环
解析虽然这是一个无符号整形i但是打印的时候用的%d(打印有符号整形的所以它是由符号位的但有人可能要问了-1不就不满足i>=0吗别忘了i是一个我无符号整型只不过打印用的%d但这并不影响i是一个无符号整型所以判断的时候i>=0是根据无符号数判断的
在这里插入图片描述

3.例三

#include<stdio.h>
int main()
{
	char a[1000];
	int i;
	for (i = 0; i < 1000; i++)
	{
		a[i] = -1 - i;
	}
	printf("%d", strlen(a));
	return 0;
}

答案255
解析经过循环a[i]里面存的是-1 -2 -3…-127 -128但是char 类型所以不存在-129 -130…-1000-128减去1就变成了127 依次往后127 126 125…3 2 1 0这才是256个数字了所以接着循环3 2 1 0…-128 -127-126…127 …3 2 1 0直到数组填满但是strlen是求字符串的长度找的是\0\0的ASCII码值是0所以到第一次为0的时候就找到了长度是255.
在这里插入图片描述

在这里插入图片描述

4.例四

#include<stdio.h>
unsigned char i = 0;
int main()
{
	for (i = 0; i <= 255; i++)
	{
		printf("hello world\n");
	}
	return 0;
}

答案死循环
解析
因为无符号char的范围是0~255所以当i=255是再++不会变成256因为会发生截断再次变成0由此造成死循环。下面是256的二进制位可以看到截断后二进制是00000000所以不会出现266条件恒成立。
在这里插入图片描述
在这里插入图片描述

二、浮点型在内存中的储存

1.浮点数

浮点数的家族包括float, double ,long double
浮点数表示范围float.h中定义

2.浮点数存储

1.二进制浮点数

根据国际标准IEEE(电气和电子工程协会)754任意一个二进制浮点数v可以表示成下面的形式:

(1). (-1)^S* M* 2^E
(2).(-1)^S表示符号位当S=0V为正数;当S1为负数。M表示有效数字大于等于1小于2’。
(3).2^E表示指数位。
v = (-1)^ s * 2^E * M

举例来说根据上述公式5.5 怎么写呢
解析1111.1111我们都知道二进制位每位都有权重小数点前面的1权重是0依次往前就是0 1 2 3…;小数点后面的权重则是-1依次往后-1 -2 -3…;所以5.5的二进制可以写成101.1小数点往前移可以写成1.011*2^2前面的第一个2代表二进制而第二个2代表移动了两位加上正负号因为5.5是正数所以S=0最终可以写成V=-1 ^ 0 * 2 ^2 1.011.所以S=0M=1.011E=2

举例来说:那么5.0和-5.0怎么写呢
解析十进制的5.0写成二进制是101.0相当于1.01×2^2。那么按照上面V的格式可以得出S=0M=1.01E=2。
十进制的-5.0写成二进制是-101.0相当于-1.01x2^2。那么S=1M=1.01E=2。
举例来说那么5.3怎写呢
解析十进制的5.3 写成二进制是不能精准由后面的数凑出来所以说不能精准保存
举例来说那么0.5怎么写呢
十进制的0.5写成二进制是0.1小数点后的第一位权重是-1所以相当于2^(-1)),也就是0.5相当于1.0*2 ^-1。那么按照上面V的格式可以得出S=0M=1.0E=-1。

2.浮点数的存储规定

IEEE 754对有效数字M和指数E还有一些特别规定。

前面说过1<M<2也就是说M可以写成1.xxxxxx的形式其中xxxxxx表示小数部分。IEEE 754规定在计算机内部保存M时默认这个数的第一位总是1因此可以被舍去只保存后面的xxxxxx部分。比如保存1.01的时候只保存01等到读取的时候再把第一位的1加上去。这样做的目的是节省1位有效数字。以32位浮点数为例留给M只有23位将第一位的1舍去以后等于可以保存24位有效数字。
至于指数E情况就比较复杂。

首先E为一个无符号整数(unsigned int)这意味着如果E为8位它的取值范围为O~255;如果E为11位它的取值范围为0-2047。但是我们知道科学计数法中的E是可以出现负数的所以IEEE754规定存入内存时E的真实值必须再加上一个中间数对于8位的E这个中间数是127;对于11位的E这个中间数是1023。比如2^10的E是10所以保存成32位浮点数时必须保存成10+127=137即10001001。
float 32位的浮点型在这里插入图片描述
double 64位的浮点型
在这里插入图片描述
也就是例如5.5的存法
二进制101.1
(-1)^ 01.011 * 2^2根据规定float类型E+127就变成了129129的二进制序列10000001(E)M存的小数点后面的也就是01100000000000000000000)再加上S中存储的0就变成了01000000101100000000000000000000转换成十六进制四个二进制转换成一个十六进制位4 0 b 0 0 0 0 0
在这里插入图片描述

3.浮点数的取出规定

然后指数E从内存中取出还可以再分成三种情况:

1.E不全为0或不全为1
这时浮点数就采用下面的规则表示即指数E的计算值减去127(或1023),有效数字M前加上第一位的1。
比如:
0.5 (1/2)的二进制形式为0.1由于规定正数部分必须为1即将小数点右移1位则为1.0*2N(-1)其阶码为-1+127=126表示为01111110而尾数1.0去掉整数部分为0补齐0到23位00000000000000000000000则其二进制表示形式为:0 01111110 00000000000000000000000

2.E全为0
这时浮点数的指数E等于1-127(或者1-1023即为真实值有效数字M不再加上第一位的1而是还原为0.xxxxxx的小数。这样做是为了表示±0以及接近于0的很小的数字。

3.E全为1
这时如果有效数字M全为0表示±无穷大(正负取决于符号位s);

3.例题

在这里插入图片描述
解析

以整数的视角存放整型的数字
00000000000000000000000000001001-9的原反补相同9(%d打印整形
浮点数存储
0(S) 00000000(E) 0000000000000000001001(M)E为全0浮点数的指数E等于1-127(或者1-1023即为真实值E=1-127=-126有效数字M不再加上第一位的1而是还原为0.xxxxxx的小数M=0.0000000000000000001001
(-1)^ 0 * 0.0000000000000000001001
2^-126所以浮点数打印出来是0.000000

以浮点数的视角存放浮点型的数字
二进制1001.0
1.001
2^3
(-1)^ 0 * 1.001 * 2^3
S=0;E=3;M=1.001
*
01000001000100000000000000000000-正数原码反码补码相同%d打印出来是1091567616
在这里插入图片描述


总结

请添加图片描述
Ending,今天的C—数据的储存下内容就到此结束啦~如果后续想了解更多就请关注我吧一键三连哦 ~

阿里云国内75折 回扣 微信号:monov8
阿里云国际,腾讯云国际,低至75折。AWS 93折 免费开户实名账号 代冲值 优惠多多 微信号:monov8 飞机:@monov6