当前位置: 首页 > news >正文

【C语言】浮点型数据在内存中的存储

🚀🚀🚀 如果文章对你有帮助不要忘记点赞关注收藏哦🚀🚀🚀

文章目录

  • ⭐浮点数在内存中的存储
    • 1.1 🤓举个例子:
    • 1.2浮点数存储规则🌈:
    • 对于M与E有一些特别规定
    • 1.3解释前面题目:
    • 完结


⭐浮点数在内存中的存储

float、double、long double

浮点数的范围定义在float.h头文件中

1.1 🤓举个例子:

#define _CRT_SECURE_NO_WARNINGS 1
#include <stdio.h>int main()
{int num = 5;float* pfloat = (float*)&num;printf("num的值为:%d\n", num);printf("*pFloat的值为:%f\n", *pfloat);*pfloat = 9.0;printf("num的值为:%d\n", num);printf("*pFloat的值为:%f\n", *pfloat);return 0;
}

输出结果

在这里插入图片描述

​ 这里肯定会有小伙伴好奇,int于float都是四个字节,为什么只是改变&num的指针类型,用%d的方式取出来值没有变化,用%f读取出来 * pfloat的值也就是num的数值会是0??而用浮点数给 * pfloat重新赋值用%d打印又不对,用%f打印出来又正常。

​ 我们都知道改变了指针类型,里面的二进制数据是不会变的,只不过读取大小和读取方式有区别,而int于float又都是四个字节,那就证明,浮点型的存取方式与整形不相同。

1.2浮点数存储规则🌈:

如果说浮点数与整数在内存中的存取方式不相同,那么浮点数的存储规则是什么呢 ?

实际上浮点数是以科学计数法的方式存储在内存中

  • (-1)^ S * M * 2^E (// 此处的^均代表指数,而不是异或)
  • S代表符号位,由于内存中存储的是二进制所以S只有0和1,0代表正数(-1的0次幂为1),1代表负数(-1的1次幂为-1)
  • M代表有效数字 大于1小于2
  • E代表 2的指数位

🌞IEEE 754标准规定,32位浮点数最高的1位是符号位s,接下来8位是指数E,剩下的23位为M.

在这里插入图片描述

64位浮点数最高的1位是符号位s,接下来11位是指数E,剩下的52位为M。

在这里插入图片描述

对于M与E有一些特别规定

​ 标准规定,对于M,由于一定是1.XXXXX,所以可以舍去第一位,取出的时候加上即可,所以存在M中的数据就是1.后面的数字,这样做可以节省一位数字,虽然M只有23位,但是节省一位之后就可以存储相当于24位的数字了。

​ 对于E,E是一个无符号整数,拿单精度浮点型float举例,E为8位,所以它的取值范围为0-255,但是科学计数法中E是可以出现负数的,,所以IEEE754标准规定,存入内存时E必须加上一个中间数,对于八位的E,这个中间数是127,对于11位的E这个中间数是1023,比如单精度E为10,那么在存储的时候,将E加上127,保存在内存中就是10001001。

​ 以上为E的存储情况,E取出的时候还分为三种情况:

E不为全0或者不为全1

这时浮点数采用下面的规则,指数E减去127(或者1023),得到真实E的值,再将M的前面加回第一位的1.

E为全0

这时,浮点数的指数E=1-127(或者1-1023)就是真实值

有效数字M不再加上前面的第一位1,这样做是为了表示±0,就是一个接近于0的一个很小的数字

E为全1

相当于±无穷大(对于float本身来说)

ps:

实际上浮点数是以科学计数法的方式存储在内存中,S、E、M是科学计数法的不同部分

​ 比如存储一个float类型110.75,计算机中存储的是二进制,转化为二进制数据1101110.11用科学计数法就表示为1.10111011*2^6

其中 符号就代表了S,S=0,1.10111011代表了M,6代表了E,八位二进制为00000110,将E加上127,将m去掉第一位1,所以这个数据存储在内存中应该就是0100 0010 1101 1101 1000 0000 0000 0000,转化为16进制为42DD 8000
让我们验证一下:

在这里插入图片描述
完全正确✨,至于数据为什么是倒着存,这就涉及到大小端字节序问题,请参考这篇博文http://t.csdn.cn/mHrtN

1.3解释前面题目:

int main()
{int num = 5;float* pfloat = (float*)&num;printf("num的值为:%d\n", num);				//5printf("*pFloat的值为:%f\n", *pfloat);		//0.000000*pfloat = 9.0;printf("num的值为:%d\n", num);				//1091567616printf("*pFloat的值为:%f\n", *pfloat);		//9.000000return 0;
}
  • 为什么num 为5,&num转换为浮点型指针就变成了0.000000? 🧐

5的二进制 00000000 00000000 00000000 00000101

转换为浮点型指针进行读取,采用浮点型的规定去读,此时在浮点型看来

在这里插入图片描述
(-1)^ S * M * 2 ^ E = (-1)^ 1 * 0.000000000000000000000101 * 2 ^ (1-127) ( 因为E全为0,参考上面E全为0取出。)

得到一个超级小的数字,对于float来说就根无穷小没有区别,截断等于0.000000。

  • 为什么对*pfloat赋值9.0,用%d输出是1091567616?🤓

将9.0的二进制是1001 科学技术法 1.001*2^3 S=0,E=3,M=1001

用float的方式存储就是 0 10000010 00100000000000000000000

用%d的方式输出,也就是用整数方式把数据读取出来,结果就是1091567616
在这里插入图片描述

最后的pfloat = 9.0用%f输出,用浮点型方式存,用浮点型方式取,自然就是9.000000了。


完结

创作不易,还请各位小伙伴多多点赞👍关注✨收藏⭐

请添加图片描述

http://www.lryc.cn/news/15026.html

相关文章:

  • impala中的刷新元数据和刷新表
  • Vscode创建vue项目的详细步骤
  • 如何在面试中介绍自己的项目,才能让软件测试面试官无可挑剔,
  • 虹科方案|从 uCPE 到成熟的边缘计算平台
  • 计算机是怎么读懂C语言的?
  • hadoop入门介绍及各组件功能运行关系
  • (HP)新手引导使用react-shepherd
  • 数据结构:栈和队列(Leetcode20. 有效的括号+225. 用队列实现栈+232. 用栈实现队列)
  • 22.2.19周赛双周赛(贪心、记忆化搜索...)
  • 2023最新软件测试面试题(带答案)
  • 【C++】类型转换方法
  • 100亿级订单怎么调度,来一个大厂的极品方案
  • C++性能白皮书
  • 华为OD机试 - 黑板上色 | 机试题算法思路 【2023】
  • 如何在六秒内吸引观众的注意力
  • FreeRTOS与UCOSIII任务状态对比
  • 小程序 npm sill idealTree buildDeps 安装一直没反应
  • GPT系列详解:初代GPT
  • 为什么要使用数据库
  • 【单目标优化算法】海鸥优化算法(Matlab代码实现)
  • 筑基六层 —— 整型提升及实用调式技巧
  • 后端前端文件传输2中传出模式
  • 【ZOJ 1067】Color Me Less 题解(vector+开方)
  • 凌恩生物经典文章:孟德尔诞辰200周年,Nature Genetics礼献豌豆高质量精细图谱
  • 进程间通信(二)/共享内存
  • 电路模型和电路定律——“电路分析”
  • 软件工程 | 第一章:软件工程学概述
  • 前端开发页面HEAD作用
  • CSS开发技巧——行为技巧
  • PX4之代码结构