浮点数在内存中的存储以及用指针改变内存与强制转换的区别

简介: 浮点数在内存中的存储以及用指针改变内存与强制转换的区别

浮点型在内存中的存储

引例

  • 我们先来看下面一段代码
#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("n的值为:%d\n", n);
  printf("*pFloat的值为:%f\n", *pFloat);
  return 0;
}
  • 可能大部分小伙伴都会认为,打印的结果分别是:
n的值为:9
pFloat的值为:9.000000
n的值为:9
*pFloat的值为:9.000000
  • 实际上,运行得到的结果为:
n的值为:9
pFloat的值为:0.000000
n的值为:1091567616
*pFloat的值为:9.000000
  • 可以看到,操作float* pFloat = (float*)&n,我们将整型n以浮点数的形式存入float型指针pFloat中,然后发现printf("pFloat的值为:%f\n", *pFloat)打印结果错误
  • 当我们执行*pFloat = 9.0这一操作,即向整型n中存入float型的值,发现printf("n的值为:%d\n", n)打印结果错误。
  • 由此我们可以知道,在内存中,浮点数和整数的存储形式是不一样的,那具体有哪些差别呢?

浮点数的表示形式

  • 根据国际标准IEEE(电器和电子工程协会)754任意一个二进制浮点数v都可以表示为这个形式:(-1)^S * M * 2 ^ E
  • (-1) ^ S表示符号位,当S为0时表示正数,当S为1时表示负数。
  • M表示有效数字,大于等于1,小于2
  • 2 ^ E表示指数位
  • 举个例子:
  • 例如十进制浮点数5.5
  • 其二进制形式为:101.1(如果不会怎么转换,请看二进制、八进制、十六进制与十进制的相互关系
  • 再用科学计数法表示:1.011 * 22
  • 加上符号位:(-1)0 * 1.011 * 22
  • 即这里的S = 0,M = 1.011,E = 2
  • 因此,对浮点数进行存储时,只需要对S,M,E这三个数进行存储就行了

浮点数的存储

  • IEEE 754规定:
  • 对于32位的浮点数(float型),最高的1位是符号位S,接着的8位是指数E,剩下的23位是有效数字M

  • 对于64位的浮点数(double型),最高的1位是符号位S,接着的11位是指数E,剩下的52位是有效数字M

  • IEEE 754对有效数字M和指数E还有些特殊的规定:
  • 对于有效数字M:
  • 之前说过,1<=M<2因此M总是可以写成1.XXXXX这样的形式
  • 754规定,在计算机内部保存有效数字M时,默认这个数的第一位总是1,因此可以被舍去,只保留后面的xxxxxx部分。
  • 等到读取的时候,再把第一位的1加上去。
  • 这样做,就可以节省一位有效数字,以double型为例,原本M为52,只能存储52为有效数字,但舍去第一位后,就相当于可以存储53位数字了。
  • 对于指数E:
  • 首先我们需要知道,指数E为一个无符号整数(unsigned int)
  • 以float型为例,E所能表示的范围是0~255。但问题是,在实际的表示中,指数E也可能存在负数的情况,例如十进制浮点数0.5,转换为二进制为0.1,则S = 1,E = -1,M = 1,那么怎么处理E为负数的情况呢?
  • IEEE 754规定,存入内存时E的真实值必须再加上一个中间数,对于8位的E,这个中间数是127,对于11位的E,这个中间数是1023。例如对于十进制浮点数0.5,存入内存时,E的值应该是-1 + 127 = 126
  • 将指数E从内存取出要分以下三种情况:
E不全为零且E不全为1
  • 这时,指数E的真实值就是存储值减去中间值(127/1023)
  • 有效数字M需要加上第一位的1。
E全为0
  • 这时,指数E的真实值就是(1 - 127) / (1 - 1023)(标准规定)
  • 有效数字M不需要加上第一位的1,而是还原为0.xxxxxxx的小数这样做是为了表示±0,以及几近于0的极小的数字(标准规定)
E全为1
  • 这时,有效数字M全为0,表示±∞(正负取决于符号位)(标准规定)

Eg

对于十进制浮点数5.5

十进制float型 -> 5.5
二进制 -> 101.1
二进制科学技术 -> 1.011 * 2 ^ 2
加上符号位 -> (-1) ^ 0 * 1.011 * 2 ^ 2
S = 0, M = 1.011, E = 2
存入到二进制序列:0 10000001 01100000000000000000000
转换为十六进制:4 0 B 0 0 0 0 0

总结

  • 我们再来回顾上面引例的代码:
#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("n的值为:%d\n", n);
  printf("*pFloat的值为:%f\n", *pFloat);
  return 0;
}
  • 整数9在内存中的存储形式为0000 0000 0000 0000 0000 0000 0000 1001
  • 我们用float类型的指针pFloat指向这段整型内存,那么根据上面的知识,我们可以得到:

  • 指数E全为0,因此printf("pFloat的值为:%f\n", *pFloat);打印出来的结果就是pFloat的值为:0.000000
  • 之后,将这段内存改为浮点数9.0
  • 9.0的二进制序列为:1001.0
  • 写成科学计数法为:(-1) ^ 0 * 1.001 * 2 * 3,即S = 0, E = 3, M = 1.001(存入内存时,E = 127 + 3 = 130,M = 0.001),即:0 10000010 00100000000000000000000

  • 如果我们用读取整数的方法来读取这段内存,那结果就是1,091,567,616也就是printf("n的值为:%d\n", n)的打印结果

用指针改变内存和强制转换的区别

  • 通过对浮点数在内存中如何储存的了解,我们应该知道,用指针改变内存的读取和强制转换变量类型是两个截然不同的操作
  • 强制转换类型:
float num_1 = 3.14;
int num_2 = (int)num_1;
printf("%d\n",num_2);
/*
  将float强制转换为int只是简单的对小数点之后的数据进行舍去,而没有改变对内存的处理
  打印结果为3.
*/
  • 指针改变内存
float num_1 = 9.5;
int *num_2 = (int *)&num_1;
printf("%d\n",*num_2);
/*
  用读取整型数据的方式读取float型的内存
  打印结果为1092091904
*/


相关文章
|
2月前
|
存储 C语言 计算机视觉
在C语言中指针数组和数组指针在动态内存分配中的应用
在C语言中,指针数组和数组指针均可用于动态内存分配。指针数组是数组的每个元素都是指针,可用于指向多个动态分配的内存块;数组指针则指向一个数组,可动态分配和管理大型数据结构。两者结合使用,灵活高效地管理内存。
|
2月前
|
容器
在使用指针数组进行动态内存分配时,如何避免内存泄漏
在使用指针数组进行动态内存分配时,避免内存泄漏的关键在于确保每个分配的内存块都能被正确释放。具体做法包括:1. 分配后立即检查是否成功;2. 使用完成后及时释放内存;3. 避免重复释放同一内存地址;4. 尽量使用智能指针或容器类管理内存。
|
18天前
|
存储 数据可视化 C++
第九问:能否尽可能详细阐述指针和引用的区别?
在C++中,指针和引用是两个重要的概念,用于操作内存地址和数据。指针是一个存储内存地址的变量,可以动态分配和释放内存;引用是变量的别名,绑定后不可改变指向。指针提供更大的灵活性和控制力,适用于复杂内存操作;引用更直观,适合简化代码并提高可读性。根据实际需求选择合适的工具。
28 0
|
2月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
152 4
|
3月前
|
存储 安全 编译器
在 C++中,引用和指针的区别
在C++中,引用和指针都是用于间接访问对象的工具,但它们有显著区别。引用是对象的别名,必须在定义时初始化且不可重新绑定;指针是一个变量,可以指向不同对象,也可为空。引用更安全,指针更灵活。
|
3月前
|
存储 C语言
数据在内存中的存储方式
本文介绍了计算机中整数和浮点数的存储方式,包括整数的原码、反码、补码,以及浮点数的IEEE754标准存储格式。同时,探讨了大小端字节序的概念及其判断方法,通过实例代码展示了这些概念的实际应用。
148 1
|
3月前
|
存储 Rust C#
内存指针解引用
【10月更文挑战第14天】
44 1
|
3月前
|
存储
共用体在内存中如何存储数据
共用体(Union)在内存中为所有成员分配同一段内存空间,大小等于最大成员所需的空间。这意味着所有成员共享同一块内存,但同一时间只能存储其中一个成员的数据,无法同时保存多个成员的值。
|
3月前
|
存储 C语言
C语言指针与指针变量的区别指针
指针是C语言中的重要概念,用于存储内存地址。指针变量是一种特殊的变量,用于存放其他变量的内存地址,通过指针可以间接访问和修改该变量的值。指针与指针变量的主要区别在于:指针是一个泛指的概念,而指针变量是具体的实现形式。
|
3月前
|
存储 弹性计算 算法
前端大模型应用笔记(四):如何在资源受限例如1核和1G内存的端侧或ECS上运行一个合适的向量存储库及如何优化
本文探讨了在资源受限的嵌入式设备(如1核处理器和1GB内存)上实现高效向量存储和检索的方法,旨在支持端侧大模型应用。文章分析了Annoy、HNSWLib、NMSLib、FLANN、VP-Trees和Lshbox等向量存储库的特点与适用场景,推荐Annoy作为多数情况下的首选方案,并提出了数据预处理、索引优化、查询优化等策略以提升性能。通过这些方法,即使在资源受限的环境中也能实现高效的向量检索。