C语音进阶————数据在内存中的存储2

2026年1月27日学习总结:

本次学习主要围绕数据在内存中的存储的浮点数的存储部分,主要分为以下三个部分:

首先在进入本次学习的内容之前先复习一下浮点数的相关知识点:

①常见浮点数:3.14159,9.18;

②浮点数家族包括:float、double、long double类型;

③浮点数表示的范围:float.h中定义;整数表示的范围在:limits.h中定义。

1.一个例子:

cs 复制代码
#include<stdio.h>
int main()
{
	int n = 9;
	float* nFloat = (float*)&n;
	printf("n的值为%d\n", n);  //9
	printf("*nFloat的值为%f\n", *nFloat);  //0.000000  
	*nFloat = 9.0;
	printf("n的值为%d\n", n);  //1091567616   修改成浮点数类型的数据,整数去访问的值则不准确
	printf("*nFlost的值为%f\n", *nFloat);  //9.000000 
	return 0;
}

运行结果:

这个例子不仅告诉我们整型和浮点数的存储方式不同,数据处理方式也不同。也告诉我们存什么值就用什么类型。

2.浮点数存储规则

n和*nFloat在内存中明明是同一个数,为什么浮点数和整数的解读结果会差这么多?要理解这个结果,一定要搞懂浮点数在计算机内部的表示方法。

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

(-1)^S*M*2^E

(-1)^S表示符号位,当S=0,V为正;当S=1,V为负。

M表示有效数字,大于等于1,小于2。

2^E表示指数位

例如:十进制的5.0,写成二进制是101.0,相当于1.01*2^2。

小数在内存中可能无法精确保存,如V=9.6=1001.100......0000,所以会出现我们输入的数可能无法一样。

(1)IEEE 754的存储规定:

①对于32位的浮点数float,最高的1位为符号位S,接着的8位是指数E,剩下的23位为有效数字M。

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

(2)IEEE 754对有效数字M和指数E还有一些特别规定:

①规定在计算机内部保存M时,默认这个数第一位总是1,因此可以被舍去,只保存后面的剩余部分。如:保存1.01时,只保存01,读数时才加上1。

目的:节省一位有效数字。以32位为例,留给M只有23位,但第一位舍去后就有24位有效数字。

②对于指数E就比较复杂

首先,E为一个无符号整数(unsigned int),8位时取值范围:0~255;11位时取值范围:0~2047。但是实际上E是可以出现负数的,所以规定存入内存的真实值必须加上一个中间数,8位的中间数是127,11位中间数是1023。

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

a.E不全为0或者不全为1:指数E的计算值减去127(或者1023),得到真实值,再将数字M前加上第一位的1。

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

c.E全为1:如果有效数字M全为0,表示±无穷大(正负取决于符号位S)。

最后再对上面示例的值进行解释:

cs 复制代码
#include<stdio.h>
int main()
{
	int n = 9;
	//原、反、补码:00000000 00000000 00000000 10010000
	float* nFloat = (float*)&n;
	//用浮点数的存储规则去读,S=0,E=00000000,M=0000000 00000000 10010000
	//所以有效值为0.0000000 00000000 10010000
    //按照读取规则精确到小数点后6为就为0.000000
	printf("n的值为%d\n", n);  //9
	printf("*nFloat的值为%f\n", *nFloat);  //0.000000  
	*nFloat = 9.0;
	//用浮点数表示时:1.001*3
	//符号位S=0,指数E=3,有效数M=1.001
	//存储为:0 10000011 00100000000000000000000而用整数来读取就为1091567616
	printf("n的值为%d\n", n);  //1091567616   
    //修改成浮点数类型的数据,整数去访问的值则不准确
	printf("*nFlost的值为%f\n", *nFloat);  //9.000000 
	return 0;
}
相关推荐
weixin_499771555 分钟前
C++中的组合模式
开发语言·c++·算法
初级代码游戏6 分钟前
套路化编程 C# winform 自适应缩放布局
开发语言·c#·winform·自动布局·自动缩放
_waylau9 分钟前
鸿蒙架构师修炼之道-架构师的职责是什么?
开发语言·华为·harmonyos·鸿蒙
2的n次方_20 分钟前
CANN Ascend C 编程语言深度解析:异构并行架构、显式存储层级与指令级精细化控制机制
c语言·开发语言·架构
iAkuya36 分钟前
(leetcode)力扣100 62N皇后问题 (普通回溯(使用set存储),位运算回溯)
算法·leetcode·职场和发展
近津薪荼36 分钟前
dfs专题5——(二叉搜索树中第 K 小的元素)
c++·学习·算法·深度优先
xiaoye-duck38 分钟前
吃透 C++ STL list:从基础使用到特性对比,解锁链表容器高效用法
c++·算法·stl
松☆41 分钟前
CANN与大模型推理:在边缘端高效运行7B参数语言模型的实践指南
人工智能·算法·语言模型
java干货1 小时前
为什么 “File 10“ 排在 “File 2“ 前面?解决文件名排序的终极算法:自然排序
开发语言·python·算法
_F_y1 小时前
C语言重点知识总结(含KMP详细讲解)
c语言·开发语言