【C语言】你知道浮点数是怎么存储的吗?(下)

简介: 【C语言】你知道浮点数是怎么存储的吗?(下)

如果你上面的图都能看懂那么下面的也就很简单了

如果我们明白了这个道理,那么我们就直接试一试吧!尝试将9.0的S,M,E写出

我们知道了方法就比较简单了,虽然刚开始可能不太熟练,见得多了也就会了。

  1. 首先我们看到是正数所以S=0
  2. 9.0转换为2进制1001.09的二进制。

3.将1001.0用类似科学计数法的形式转换得到

1.001 * 2 ^ 3,我们得到M=1.001 ,E = 3。

这三步做好了就基本不会出错了。

但是我们还要讲一个比较重要的概率,就是:

既然小数位也是用2 ^ -1, 2 ^-2,·······2 ^ n 来逐渐趋近数据的小数点后的值

那么3.14这个值呢?

我们再用我们的方法试试是搞不出来的。

下面是我用叫AI算的

11.00100110...(无限循环,以省略号表示)

0.14这样的数需要很多小数点后的1来表示,但是我们计算机只会保留部分,所以会导致精度丢失,但是没有办法,所以我们才会四舍五入。

知道了这些知识我们再来看看到底计算机能存储多少有效的数位呢?

IEEE 754规定:

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

IEEE 754规定:

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

由于52比特位实在是太长了,我就换了一下行,但是实际上是连起来的。

M与E的特别规定🐣

  1. M的相关规定

在上面我们说过M的范围为[1, 2)。也就是说M最小也是1。

IEEE 754规定,当计算机内部保存M的时候,会默认这个数第一位总是1,

因此可以被舍去,只保存小数点后面部分。

这样做的好处是,当省略掉一个比特位,那么就多出一个比特位用来增加精度。

比如保存1.01的时 候,只保存01,等到读取的时候,再把第一位的1加上去。
这样做的目的,是节省1位有效数字。
以32位 浮点数为例,留给M只有23位, 将第一位的1舍去以后,等于可以保存24位有效数字。

  1. E的相关规定

指数E的情况是比较复杂的。

  • 我们看到E有8位,并且这8位是无符号的,也就是说,E最大是1111 1111
    就是255,最小为0; 0 < = E < = 255(32比特位浮点数)。
  • 如果E为11位,它的取值范围为0~2047

但是,我们知道,科学计数法中的E是可以出现负数的。

那该怎么办?

所以IEEE 754规定,存入内存时E的真实值必须再加上一个中间数,对于8位的E,这个中间数是127;

对于11位的E,这个中间数是1023。

比如,2^10的E是10,所以保存成32位浮点数时,必须保存成10+127=137,即1000 1001。

我们来看两个例子:

然后我们调试起来验证验证~

代码也给出来

int main()
{
    float f = 5.5;
    //101.1
    // S = 0,M = 1.011,E=2
    //-1^(0) * 1.011 * 2^2
    //当要存储时E+127
    // S  E              M
    //0 10000001 01100000000000000000000
    //0100 0000 1011 0000 0000 0000 0000
    //40  b0 00 00
    //由于是小端存储
    //00 00 0b 40
    return 0;
}

我们再换当E不加127时是负数的例子

int main()
{
    float f = 0.5;
    //0.1
    //S = 0, M = 1.0 E = -1
    //-1*(0) * 1 * 2^-1
    //E+127 = 126
    //0 01111110 0000000000000000000000
    //0011 1111 0000 0000 0000 0000
    //3f 00 00 00
    // 00 00 00 3f
    return 0;
}

如果说把数据放进去需要根据是32位还是64位来决定放的是多大的E,

那么取出来的时候是不是也有很多情况呢?

确实是这样!

指数E从内存中取出还可以分成三种情况🌑

  1. E不全为0或不全为1

这时,浮点数就采用下面的规则表示,即指数E的计算值减去127(或1023),

得到真实值,再将有效数字M前加上第一位的1。

比如:

0.5(1/2)的二进制形式为0.1,由于规定正数部分必须为1,

即将小数点右移1位,则为1.0*2^(-1),其阶码为-1+127=126,

表示为01111110,而尾数1.0去掉整数部分为0,

补齐0到23位00000000000000000000000,则其二进制表示形式为:

0 01111110 00000000000000000000000

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

为什么会被直接解析为0呢? 因为 2^-126次方已经非常小了,相当与趋近余0,地球离月球的距离约为2的28.8974次方,这样对比就知道2 ^-126 有多小了吧。

  1. E全为1

这时,如果有效数字M全为0,

表示±无穷大(正负取决于符号位s);


为什么呢?

我们知道E最大是1111 1111也就是255,

255 - 127 也还有128。

2^128次方是多大?

是不是就相当于-+无穷!

我们回到引例🥶

我们现在再来看第二行和第三行。

1.第二行解析:

2. 第三行解析

就是这么奇妙!

总结😈

我们同过这篇博客系统的学习了,二进制与十进制在科学计数法上的不同,也了解的浮点数在内存中存储方式,和注意的要点,并且我们最后也帮大家解答了最开始的题目

最后如果这篇博客有帮助到你,欢迎点赞关注加收藏

如果本文有任何错误或者有疑点欢迎在评论区评论


目录
相关文章
|
2月前
|
存储 编译器 C语言
C语言存储类详解
在 C 语言中,存储类定义了变量的生命周期、作用域和可见性。主要包括:`auto`(默认存储类,块级作用域),`register`(建议存储在寄存器中,作用域同 `auto`,不可取地址),`static`(生命周期贯穿整个程序,局部静态变量在函数间保持值,全局静态变量限于本文件),`extern`(声明变量在其他文件中定义,允许跨文件访问)。此外,`typedef` 用于定义新数据类型名称,提升代码可读性。 示例代码展示了不同存储类变量的使用方式,通过两次调用 `function()` 函数,观察静态变量 `b` 的变化。合理选择存储类可以优化程序性能和内存使用。
152 82
|
27天前
|
存储 C语言
深入C语言内存:数据在内存中的存储
深入C语言内存:数据在内存中的存储
|
2月前
|
存储 人工智能 C语言
数据结构基础详解(C语言): 栈的括号匹配(实战)与栈的表达式求值&&特殊矩阵的压缩存储
本文首先介绍了栈的应用之一——括号匹配,利用栈的特性实现左右括号的匹配检测。接着详细描述了南京理工大学的一道编程题,要求判断输入字符串中的括号是否正确匹配,并给出了完整的代码示例。此外,还探讨了栈在表达式求值中的应用,包括中缀、后缀和前缀表达式的转换与计算方法。最后,文章介绍了矩阵的压缩存储技术,涵盖对称矩阵、三角矩阵及稀疏矩阵的不同压缩存储策略,提高存储效率。
345 8
|
1月前
|
存储 C语言
C语言中的浮点数存储:深入探讨
C语言中的浮点数存储:深入探讨
|
2月前
|
存储 算法 C语言
数据结构基础详解(C语言): 二叉树的遍历_线索二叉树_树的存储结构_树与森林详解
本文从二叉树遍历入手,详细介绍了先序、中序和后序遍历方法,并探讨了如何构建二叉树及线索二叉树的概念。接着,文章讲解了树和森林的存储结构,特别是如何将树与森林转换为二叉树形式,以便利用二叉树的遍历方法。最后,讨论了树和森林的遍历算法,包括先根、后根和层次遍历。通过这些内容,读者可以全面了解二叉树及其相关概念。
|
2月前
|
存储 机器学习/深度学习 C语言
数据结构基础详解(C语言): 树与二叉树的基本类型与存储结构详解
本文介绍了树和二叉树的基本概念及性质。树是由节点组成的层次结构,其中节点的度为其分支数量,树的度为树中最大节点度数。二叉树是一种特殊的树,其节点最多有两个子节点,具有多种性质,如叶子节点数与度为2的节点数之间的关系。此外,还介绍了二叉树的不同形态,包括满二叉树、完全二叉树、二叉排序树和平衡二叉树,并探讨了二叉树的顺序存储和链式存储结构。
|
2月前
|
存储 算法 C语言
C语言手撕数据结构代码_顺序表_静态存储_动态存储
本文介绍了基于静态和动态存储的顺序表操作实现,涵盖创建、删除、插入、合并、求交集与差集、逆置及循环移动等常见操作。通过详细的C语言代码示例,展示了如何高效地处理顺序表数据结构的各种问题。
|
2月前
|
存储 缓存 程序员
c语言的存储类型-存储类
本文详细介绍了C语言中的存储类型及其分类,包括基本类型(如整型、浮点型)和复合类型(如数组、结构体)。重点讲解了不同存储类别(`auto`、`static`、`register`、`extern`、`typedef`、`volatile`、`const`)的特点及应用场景,并展示了C11/C99引入的新关键字(如`_Alignas`、`_Atomic`等)。通过示例代码解释了每个存储类别的具体用法,帮助读者更好地理解和运用这些概念。
|
3月前
|
存储 API C语言
【C语言】 作用域和存储期
【8月更文挑战第18天】
|
3月前
|
存储 C语言
【C语言】进程间通信之存储映射区mmap
【C语言】进程间通信之存储映射区mmap
40 0