关于整数和浮点数在内存中的存储的总结

简介: 关于整数和浮点数在内存中的存储的总结

前言:

首先必须要说明的是:整数和浮点数在内存中的存储是不同的,有人或许会问,都是二进制码有何不同呢?

1.整数的存储:

整数在数据中的存储是以二进制的形式进行,即0和1的形式进行,我们都知道,整数的数据类型是整型,整型对应四个字节,一个字节8个比特位,故一个整型32个比特位,我们这里所说的比特位就是0或者1,比如,我们把数字1存储在内存中是00000000000000000000000000000001。

其次,整数存储在内存中是以补码的形式存储的,对于大于等于0的数字来说,原码等于反码等于补码,而对于小于0的数字来说,原码,反码,补码是三个独立的不完全相同的三个二进制码,其关系为:

原码符号位不变,其他位置按位取反=反码,反码+1=补码

补码得到原码,即把整个过程反过来即可。

2.浮点数的存储:

浮点数得存储与整数是完全不同的,但我将其分为SEM存储法,浮点数的存储遵循的规律是:(-1)^S *M 2 ^E 。
1.S代表此浮点数的符号,负数则S等于1,否则等于0
2.M为真实的数字,范围在1到2之间但不等于2
3.E为指数
假如我们拿出来一个数字为9.25,我们首先要将其写成二进制形式,即为1001.001,由M的范围,我们要把这个数字变成1.001001
2^3,好,下一步我们就要把数据存储到内存中了。

由SEM存储法,且float类型也是四个字节,在32位环境下,我们要分成3个部分来存储数据,S区,E区,M区按次序划分,其中S去占一个比特位,E区在32位下是8比特位,64位下是11比特位,M区在32位下是23比特位,64位下是52比特位,如下图:

让我们接着来存储数据:9.0是正数,所以S位则为0,由于我们的M值默认为1.多,故我们直接省略1,只存储小数点后面的部分进入M区,后续加上一个1即可,这样可能让数据多存一些,故我们的M区为00100000000000000000000,对于E区,有时候我们是有可能得到负数的,这种情况下8位没法存储,所以我们这里用一个中间值来存储,32位下为127,64位下为1023,任意的E都要加上127或者1023后的结果存储在E区,所以我们这里是127+3=130,130二进制为10000100,由此,我们完整的看一看9.25存储在计算机内存中是怎样的,即为:0 10000100 00100000000000000000000.这便是浮点数的存储。

3.浮点数的读取:

浮点数的读取大体分为三种情况:

1.E区非全0也非全1:

这种情况下,E区的值-127或者1023后,按照存储的规律处理即可。

2.E区全0:

让我们想想E区全0,证明加上127或者1023后仍然是0,可证明这个E是一个很大的负数,对于这种情况,我们直接省略最后的加一步骤,直接返回M作为真实值,同时2^E的E也变成1-127即可。

3.E区全1:

E区全1,证明加上127或者1023后全1,这个数会是正负无穷大的,取决于符号。

4.存储和读取的灵活理解和转化:

int 和float都是4字节的数据类型,这决定了他们的存储二进制码是可以共用的,不像char和int那样需要整型提升和截断的问题,但由于读取方式的不同导致其会出现不同的结果。

看下面这道题:

#include <stdio.h>
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;
 }
  //运行结果为:9       0.000000         1091567616       9.000000

首先,n先以整型存储,然后以整型读取,故结果就是9

然后是以浮点型存储,按照上面的规律,E区全零,得到的结果是一个很小的数,故为0.000000

然后由解引用改为浮点型9.0,以整型读取,故为1091567616

最后以浮点型读取,故就为9.000000

通过这道题,我们就可以清楚的理解我所说的数据的存储和读取之间的灵活转化,我们处理的时候一定要看好存储和读取都是以怎样的形式进行的。

5.大小端存储:

内存中的基本单位是1字节,我们假设把一个数据类型想成一个队列,队列是需要按照一定顺序区排列的,数据也需要,比如拿出一个整型的数据,四个字节,这个字节需要一定的顺序排列,这便需要大端小端存储了。

大端存储:低地址放高位,高地址放低位

小端存储:低地址放低位,高地址放高位

例如:0X11223344

我们假设最左边为低地址,最右边为高地址

大端存储就是:0X11 22 33 44

小端存储就是:0X44 33 22 11

6.总结:

以上便是整数和浮点数在内存中的存储问题,这里面的知识点不难,只是需要记很多东西,建议忘记的时候就反复拿出来复习,强化自己对于数据类型的理解能力和内功。

目录
相关文章
|
2月前
|
存储 C语言
数据在内存中的存储方式
本文介绍了计算机中整数和浮点数的存储方式,包括整数的原码、反码、补码,以及浮点数的IEEE754标准存储格式。同时,探讨了大小端字节序的概念及其判断方法,通过实例代码展示了这些概念的实际应用。
104 1
|
2月前
|
存储
共用体在内存中如何存储数据
共用体(Union)在内存中为所有成员分配同一段内存空间,大小等于最大成员所需的空间。这意味着所有成员共享同一块内存,但同一时间只能存储其中一个成员的数据,无法同时保存多个成员的值。
|
2月前
|
存储 弹性计算 算法
前端大模型应用笔记(四):如何在资源受限例如1核和1G内存的端侧或ECS上运行一个合适的向量存储库及如何优化
本文探讨了在资源受限的嵌入式设备(如1核处理器和1GB内存)上实现高效向量存储和检索的方法,旨在支持端侧大模型应用。文章分析了Annoy、HNSWLib、NMSLib、FLANN、VP-Trees和Lshbox等向量存储库的特点与适用场景,推荐Annoy作为多数情况下的首选方案,并提出了数据预处理、索引优化、查询优化等策略以提升性能。通过这些方法,即使在资源受限的环境中也能实现高效的向量检索。
|
2月前
|
存储 编译器
数据在内存中的存储
数据在内存中的存储
44 4
|
2月前
|
存储 Java
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
这篇文章详细地介绍了Java对象的创建过程、内存布局、对象头的MarkWord、对象的定位方式以及对象的分配策略,并深入探讨了happens-before原则以确保多线程环境下的正确同步。
60 0
JVM知识体系学习四:排序规范(happens-before原则)、对象创建过程、对象的内存中存储布局、对象的大小、对象头内容、对象如何定位、对象如何分配
|
28天前
|
缓存 Prometheus 监控
Elasticsearch集群JVM调优设置合适的堆内存大小
Elasticsearch集群JVM调优设置合适的堆内存大小
232 1
|
18天前
|
存储 监控 算法
深入探索Java虚拟机(JVM)的内存管理机制
本文旨在为读者提供对Java虚拟机(JVM)内存管理机制的深入理解。通过详细解析JVM的内存结构、垃圾回收算法以及性能优化策略,本文不仅揭示了Java程序高效运行背后的原理,还为开发者提供了优化应用程序性能的实用技巧。不同于常规摘要仅概述文章大意,本文摘要将简要介绍JVM内存管理的关键点,为读者提供一个清晰的学习路线图。
|
27天前
|
Java
JVM内存参数
-Xmx[]:堆空间最大内存 -Xms[]:堆空间最小内存,一般设置成跟堆空间最大内存一样的 -Xmn[]:新生代的最大内存 -xx[use 垃圾回收器名称]:指定垃圾回收器 -xss:设置单个线程栈大小 一般设堆空间为最大可用物理地址的百分之80
|
28天前
|
Java
JVM运行时数据区(内存结构)
1)虚拟机栈:每次调用方法都会在虚拟机栈中产生一个栈帧,每个栈帧中都有方法的参数、局部变量、方法出口等信息,方法执行完毕后释放栈帧 (2)本地方法栈:为native修饰的本地方法提供的空间,在HotSpot中与虚拟机合二为一 (3)程序计数器:保存指令执行的地址,方便线程切回后能继续执行代码
21 3
|
28天前
|
存储 缓存 监控
Elasticsearch集群JVM调优堆外内存
Elasticsearch集群JVM调优堆外内存
46 1