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

news2024/9/20 22:13:38

前言
🎈大家好,我是何小侠🎈
🍃大家可以叫我小何或者小侠🍃
💐希望能通过写博客加深自己对于学习内容的理解💐
🌸也能帮助更多人理解和学习🌸

积学以储宝,酌理以富才— 出自《文心雕龙·神思》
解释:积累学识来储存珍宝,要斟酌辨析各种事理来丰富增长自己的才学。

在这里插入图片描述


    这篇博客将会详细介绍浮点数的存储,我们在很多教材上是学不到的,所以我们会对浮点数的存储产生好奇,今天我们就一起来学习。

    在这里插入图片描述

    概览

    • 常见的浮点数🍀
    • 引例🐽
    • 引例答案✳️
    • 浮点数的存储规则💥
    • M与E的特别规定🐣
      • <font color=#FF size= 4>指数E从内存中取出还可以分成三种情况🌑</font >
    • <font color= #00009C size=5>我们回到引例🥶
    • 总结😈

    常见的浮点数🍀

    例如:1.2233344
    又如:1E3
    1E3我们应该好好介绍一下:
    实际上就是我们初中物理中学习的科学计数法,
    就等于1 x 10^3 = 1000。
    还有一点规定,就是E前面的数字可以是小数,而E后面的数字只能是整数
    这与我们写物理题目是一个道理,毕竟我们从来没有都不会让小数成为
    10的指数。

    我所使用的编译器是VS2022,我们如果想知道我们编译器所规定的数据类型的最大值最小值可以查找两个头文件。

    1. limits.h是用来查看除浮点型以外的数据类型的。
      在这里插入图片描述

    在这里插入图片描述
    由于我们的重点是浮点数,就不一一展示了

    1. float.h就是用来查看编译器所设定的浮点数的最大值最小值的,但是在VS2022上很难查找到这个文件,所以我借助了everything这个很好用的查找文件的工具。

    在这里插入图片描述

    我们已经学会了如何查看头文件,接下来我们再来看一个例子。

    引例🐽

    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;
    }
    
    

    我们看看这行代码,再分析一下。
    下面是我的分析图,你也可以看看。

    • 我们先看上面部分
      在这里插入图片描述
      我们 printf("n的值为:%d\n", n);
      这行代码是比较简单的,输出9,
      那么 printf("*pFloat的值为:%f\n", *pFloat);这一行就有点不确定了,我们知道如果以%d打印,那么我们要打印的数据在%d看来就是一个有有符号整型或者以%u打印,那么要打印的数据在%u看来就是一个无符号的整型。
      这里应该也是一样的,那么以%f来打印,是打印9.000000吗?我们暂时放一放看下半部分。

    • 下半部分
      在这里插入图片描述
      我们知道*pfloat = 9.0会改变这两个变量里的数据,那么float *类型的是比较容易改变的,会被直接改变变成9.000000,但是int 类型的变量n呢?如果你不了解浮点数在内存中的存储,你确实很难知道为什么。
      我们先将答案给出来,大家带着答案一起去寻找为什么

    引例答案✳️

    在这里插入图片描述
    这真的很奇怪是不是,第二行和第三行好像是完全没有道理。但是如果你仔细一想就应该能推提出一个假设浮点数的存储方式一定与整型不同。
    别急我们马上就来介绍。

    浮点数的存储规则💥

    我们想搞懂这个问题需要我们了解浮点数在内存中的存储方法
    放心不是很复杂,只要我们用点心。

    在C语言中,浮点数的存储方式遵循IEEE 754标准。具体来说,C语言中的浮点数类型(如float、double)使用二进制表示,并按照IEEE 754标准进行存储。

    IEEE 754规定:
    任何一个二进制浮点数N都可以表达成下面的方式:

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

    • 在这个式子中(-1) ^ S表示符号位,当S为0时,N为正数,当S为1时,N为负数。
    • M表示的是有效位数,大于1,小于2。
    • 2 ^ E表示指数位。

    这样你肯定也还是不明白,没关系我们举例就行。

    首先我们也还是需要复习一下我们的数学知识:
    在这里插入图片描述
    我们更仔细的来看就是这样的:
    在这里插入图片描述
    那么如果十进制是这样那么二进制呢?

    当然是差不多的,我们再来举个例子
    在这里插入图片描述
    如果不是很懂可以看这副图在这里插入图片描述
    如果你上面的图都能看懂那么下面的也就很简单了
    在这里插入图片描述
    如果我们明白了这个道理,那么我们就直接试一试吧!尝试将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. 第三行解析

    在这里插入图片描述在这里插入图片描述

    就是这么奇妙!

    总结😈

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

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

    在这里插入图片描述在这里插入图片描述
    如果本文有任何错误或者有疑点欢迎在评论区评论
    在这里插入图片描述

    在这里插入图片描述

    本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/733538.html

    如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

    相关文章

    Debian 11 x64 安装 MySQL 8.0.33

    更新 sudo apt update sudo apt install gnupg安装 DEB Package wget -c https://dev.mysql.com/get/mysql-apt-config_0.8.25-1_all.deb sudo dpkg -i mysql-apt-config_0.8.25-1_all.deb具体版本见官方网站&#xff1a;MySQL Community Downloads&#xff0c;这里仅以版本 …

    详解什么是新零售和新零售的四种商业模式

    前言 自推出新零售概念以来&#xff0c;新零售已成为当前的热门话题。今天我们将进一步了解什么是新零售。 一、什么是新零售? 新零售&#xff0c;英文是New Retailing&#xff0c;即企业以互联网为依托&#xff0c;通过运用大数据、人工智能等先进技术手段&#xff0c;对商…

    VMware虚拟机里的Ubuntu通过主机的代理联网

    问题描述&#xff1a;主机win10&#xff0c;通过代理联网。主机里装有VMware的虚拟机Ubuntu&#xff0c;想要通过主机的代理进行上网。 步骤&#xff1a; 1 将虚拟机的网络设置为NAT模式。 2 在win10命令行中输入ipconfig&#xff0c;查询ipv4的局域网地址。&#xff08;注&…

    使用docker安装Nacos,远程连接nacos报错,please check server x.x.x.x ,port 9848 is available

    报错: please check server 127.0.0.1 ,port 9848 is available 原因: 当nacos客户端升级为2.x版本后&#xff0c;新增了gRPC的通信方式&#xff0c;新增了两个端口。这两个端口在nacos原先的端口上(默认8848)&#xff0c;进行一定偏移量自动生成.。 当客户端升级成2.x版本时&…

    [工业互联-20]:常见EtherCAT主站方案:TwinCAT的Windows 解决方案

    目录 第1章 TwinCAT简介 第2章 软件架构 第3章 应用程序架构 第1章 TwinCAT简介 TwinCAT是由德国Beckhoff公司开发的一套功能强大的自动化软件平台。 它是一个集成的开发环境&#xff0c;用于实现实时控制、PLC编程、运动控制、HMI&#xff08;人机界面&#xff09;设计和…

    service 2 暴露服务的 3种 方式

    【k8s 系列】k8s 学习十九&#xff0c;service 2 之前我们简单的了解一下 k8s 中 service 的玩法&#xff0c;今天我们来分享一下 service 涉及到的相关细节&#xff0c;我们开始吧 为什么要有 服务 Service&#xff1f; 因为服务可以做到让外部的客户端不用关心服务器的数量…

    【二叉树part09】| 669.修剪二叉搜索树、108.将有序数组转换为二叉搜索树、538.把二叉搜索树转换为累加树

    目录 &#x1f388;LeetCode669. 修剪二叉搜索树 &#x1f388;LeetCode108.将有序数组转换为二叉搜索树 &#x1f388;LeetCode538.把二叉搜索树转换为累加树 &#x1f388;LeetCode669. 修剪二叉搜索树 链接&#xff1a;669.修剪二叉搜索树 给你二叉搜索树的根节点 root…

    使用Go 语言的三个原因

    几个星期前&#xff0c;我一个朋友问我&#xff1a;“为什么要关心 Go 语言”&#xff1f; 因为他们知道我热衷于 Go 语言&#xff0c;但他们想知道为什么我认为其他人也应该关心。有三个原因&#xff1a;安全性、生产力和并发性。有些语言可以涵盖一个也有可能是两个方面&…

    代码逐行解析!冠军选手解读锂电池生产温度预测赛事方案

    Datawhale干货 作者&#xff1a;鱼佬、骆秀韬&#xff0c;Datawhale成员 本实践是数据挖掘类型的比赛&#xff0c;聚焦于工业场景。实践任务本质上为回归任务&#xff0c;其中会涉及到时序预测相关的知识。 本实践可帮助大家&#xff1a; 快速掌握数据挖掘任务基本流程&#x…

    【开源-文章迁移利器】MarkDown本地图片转云端存储脚本-支持目录递归查找转换

    从一些笔记软件导出markdown文档后&#xff0c;图片都是本地图片&#xff0c;文档数量过多&#xff0c;用typora一一打开上传图片过于繁琐&#xff0c;特开发一个一键迁移文章图片的脚本&#xff0c;方便markdown文档的迁移。 文章目录 大致需求开源地址设计思路脚本介绍快速使…

    蓝桥杯专题-真题版含答案-【大衍数列】【圆周率】【分糖果】【等额本金】

    点击跳转专栏>Unity3D特效百例点击跳转专栏>案例项目实战源码点击跳转专栏>游戏脚本-辅助自动化点击跳转专栏>Android控件全解手册点击跳转专栏>Scratch编程案例点击跳转>软考全系列点击跳转>蓝桥系列 &#x1f449;关于作者 专注于Android/Unity和各种游…

    Java csv文件上传下载中的相关转换

    目录 一. 需求二. List<Entity>转List<List<String>>2.1 实体类2.2 转换 三. 上传csv文件转List<Map>3.1 csv文件3.2 前台3.3 实体类3.4 转换3.5 效果 一. 需求 &#x1f914;项目中遇到了两个需求 1.查询数据库&#xff0c;得到List<Entity>这…

    快速搭建一个美观且易用的 Django 管理后台 —— django-xadmin

    Django-xadmin&#xff08;也称为Xadmin&#xff09;是一个第三方的 Django 应用程序&#xff0c;它提供了一系列工具和模板来快速开发基于 Django 的后台管理界面。使用 Django-xadmin 可以用很少的代码就创建出一个强大的、具备实时查看数据、增、删、改等基本操作的 Django …

    leetcode-704.二分查找

    leetcode-704.二分查找 文章目录 leetcode-704.二分查找一.题目描述二.第1次代码提交(非二分查找)三.第2次代码提交(非二分查找&#xff0c;std::find和std::distance)四.第3次代码提交(二分查找)五.关于C中int型的奇数除以2 一.题目描述 二.第1次代码提交(非二分查找) class …

    Linux 学习记录47(QT篇待完成)

    Linux 学习记录47(QT篇) 本文目录 Linux 学习记录47(QT篇)一、将资源文件加载到项目1. 将资源文件放到项目下2. 添加到项目 二、信号与槽机制1. 信号与槽机制概念2. 信号与槽概念 三、四、思维导图练习1. main_page.cpp2. main.cpp3. main_page.h4. login.cpp5. login.h 一、将…

    Delphi 11必备指南:使用Git集成Python4Delphi的完整步骤

    在Delphi中使用Python有很多好处&#xff0c;可以扩展Delphi的功能并利用Python强大的科学计算和数据分析库。但是&#xff0c;为了将Python集成到Delphi中&#xff0c;我们需要安装Python for Delphi (P4D)组件套件。在这篇博客中&#xff0c;我将介绍如何使用Git安装P4D组件套…

    N-122基于springboot,vue网上订餐系统

    开发工具&#xff1a;IDEA 服务器&#xff1a;Tomcat9.0&#xff0c; jdk1.8 项目构建&#xff1a;maven 数据库&#xff1a;mysql5.7 前端技术 &#xff1a;VueElementUI 服务端技术&#xff1a;springbootmybatisredis 本系统分用户前台和管理后台两部分&#xff0c;…

    python_day3_list

    数据容器 &#xff1a; list&#xff08;列表&#xff09; tuple&#xff08;元组&#xff09; str&#xff08;字符串&#xff09; set&#xff08;集合&#xff09; dict&#xff08;字典&#xff09; 列表 list name_list [java, c, python] print(name_list) print(type…

    【LeetCode每日一题合集】2023.7.3-2023.7.9

    文章目录 2023.7.3——445. 两数相加 II&#xff08;大数相加/高精度加法&#xff09;2023.7.4——2679. 矩阵中的和2023.7.5——2600. K 件物品的最大和&#xff08;贪心&#xff09;代码1——贪心模拟代码2——Java一行 2023.7.6——2178. 拆分成最多数目的正偶数之和&#x…

    C. Vampiric Powers, anyone? - 思维+前缀和

    分析&#xff1a; 添加新元素的操作可以理解为添加任意一段以n结尾的异或和&#xff0c;当原数组总异或和与新加的元素进行异或又可以得到剩余的前缀的异或和&#xff0c;假设新加的元素的值是i到n的异或和x&#xff0c;那么总异或和sumpre^x&#xff0c;所以sum^xpre&#xff…