C—数据的储存(下)

2023-07-16 20:35:30


前言

👧个人主页:@小沈熬夜秃头中୧⍤⃝❅
😚小编介绍:欢迎来到我的乱七八糟小星球🌝
📋专栏: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=0,V为正数;当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=0;M=1.011;E=2

举例来说:那么5.0和-5.0怎么写呢?
解析:十进制的5.0,写成二进制是101.0,相当于1.01×2^2。那么,按照上面V的格式,可以得出S=0,M=1.01,E=2。
十进制的-5.0,写成二进制是-101.0,相当于-1.01x2^2。那么,S=1,M=1.01,E=2。
举例来说那么5.3怎写呢?
解析:十进制的5.3 写成二进制是不能精准由后面的数凑出来所以说不能精准保存
举例来说那么0.5怎么写呢?
十进制的0.5,写成二进制是0.1(小数点后的第一位权重是-1所以相当于2^(-1)),也就是0.5,相当于1.0*2 ^(-1)。那么,按照上面V的格式,可以得出S=0,M=1.0,E=-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就变成了129;129的二进制序列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—数据的储存(下)内容就到此结束啦~,如果后续想了解更多,就请关注我吧,一键三连哦 ~

更多推荐

SQL server 创建存储过程

SQLServer如何创建存储过程存储过程:可以理解为完成特定功能的一组SQL语句集,存储在数据库中,经过第一次编译,之后的运行不需要再次编译,用户通过指定存储过程的名字并给出参数(如果该存储过程带有参数)来调用存储过程。如何创建存储过程创建存储过程可以有两种方式:一是在SSMS中界面操作创建存储过程,而后修改其中的S

【MediaSoup---源码篇】(二)Router

概述Router:路由对象,类似于房间的功能,保存了流之间的订阅关系,它接收Producer的数据并转发给订阅该Producer的ConsumerRouter继承了以下类:RTC::Transport::Listener,RTC::RtpObserver::Listener,Channel::ChannelSocket

全面解析缓存应用经典问题

1、前言随着互联网从简单的单向浏览请求,发展为基于用户个性信息的定制化以及社交化的请求,这要求产品需要做到以用户和关系为基础,对海量数据进行分析和计算。对于后端服务来说,意味着用户的每次请求都需要查询用户的个人信息和大量的关系信息,此外大部分场景还需要对上述信息进行聚合、过滤、排序,最终才能返回给用户。CPU是信息处理

接口自动化测试完整版(文档+视频)

1.什么是接口测试顾名思义,接口测试是对系统或组件之间的接口进行测试,主要是校验数据的交换,传递和控制管理过程,以及相互逻辑依赖关系。其中接口协议分为HTTP,WebService,Dubbo,Thrift,Socket等类型,测试类型又主要分为功能测试,性能测试,稳定性测试,安全性测试等。在分层测试的“金字塔”模型中

QT windows dpi变化导致的界面异常处理

问题:已经在UI界面中利用布局设计好界面,但是运行程序后显示的界面出现错乱,而且在做出一些修改后重新构建,运行时界面无变化。目录一、解决UI设计界面与运行时显示界面不一致的问题1、导致该现象的原因有:显示屏分辨率过高2、使用布局和设置控件sizePolicy二、解决修改界面后重构,运行无变化的问题一、解决UI设计界面与

Spring | Spring监听器使用与入门

引言在复杂的软件开发环境中,组件之间的通信和信息交流显得尤为重要。Spring框架,作为Java世界中最受欢迎的开发框架之一,提供了一种强大的事件监听器模型,使得组件间的通信变得更加灵活和解耦。本文主要探讨Spring事件监听器的原理、使用方法及其在实际开发中的应用,希望为广大开发者提供实用的参考。1.1Spring事

2023年中国研究生数学建模竞赛赛题浅析

为了更好的帮助大家第一天选题,这里首先为大家带来各个赛题的赛题浅析,分析对应赛题之后做题阶段可能会遇到的各种难点。稍后会带来D题的详细解析思路,以及相关的其他版本解题思路成品论文等资料。赛题难度评估:A、B>C>E、F>D选题人数评估:D>E、F>C>A、BA题WLAN网络信道接入机制建模B题DFT类矩阵的整数分解逼近

【Java 并发编程】CopyOnWriterArrayList 详解

CopyOnWriterArrayList详解1.ArrayList1.1ArrayList和LinkedList的区别1.2ArrayList如何保证线程安全2.CopyOnWriteArrayList原理3.CopyOnWriteArrayList的优缺点3.1优点3.2缺点4.源码分析4.1两个成员变量4.2构造

通过篡改cred结构体实现提权利用

前言在之前的HeapOverflow文章中,作者还构造了任意地址读写的操作,使用了任意地址读写去进行提权,还挺有意思的,记录一下如何利用任意地址读写进行提权。作者利用任意地址读写分别改写modprobe_path以及cred结构体去实现提权的操作,由于改写modprobe_path的方法之前已经研究过了,因此现在详细记

【Qt图形视图框架】QGraphicsScene分析

QGraphicsScene分析描述注意使用示例属性方法成员方法(较重要的)公共槽函数:信号:事件描述QGraphicsScene类提供了一个用于管理大量2D图形项的表面。该类充当QGraphicsItems的容器。它与QGraphicsView一起用于在2D表面上可视化图形项,如线条、矩形、文本,甚至自定义项。QGr

“undefined reference to XXX“问题总结

"undefinedreferencetoXXX"问题总结引言我们在Linux下用C/C++工作的时候,经常会遇到"undefinedreferencetoXXX"的问题,直白地说就是在链接(从.cpp源代码到可执行的ELF文件,要经过预处理->编译->链接三个阶段,此时预处理和编译已经通过了)的时候,链接器找不到XX

热文推荐