国际 IEEE 754 标准,为啥会有精度缺失

本贴最后更新于 1648 天前,其中的信息可能已经渤澥桑田

## 国际标准 IEEE 754 浮点数的问题

- 整数的二进制比如 2 的二进制是 10 没疑问

- 小数的二进制比如 0.125 的二进制是 001 流程如下


//每次乘以2取整数部分,每次取完整数部分保留小数部分再乘以2

0.125

0.25=0.125*2  0

0.5=0.25*2    0

1.0=0.5*2     1

0.0=0.0*2     0

0.0           0

### 问题

1. 为啥这样存有什么原因,有什么好处和坏处

2. 为啥要指数表达法

3. 为啥要 补位

4. 为啥忽略 1

好了,上面这些问题,没事就是让你知道有这种问题就行,因为希望你在看下面的例子会产生这样的疑问,最后会一一解答

### IEEE 754 的内存结构

| 数符 | 阶码 | 尾数 |

| ---- | ---- | ----| 

| 1 位 | 8 位 | 23 位 |

整体的结构就是

0 00000000 00000000000000000000000 (问题1)

### 2.125 的内存二进制表达

1. 整数位 2 获取到它的二进制为 10

2. 0.125 的二进制表达为 001

3. 合并 2 个二进制就是 10.001

4. 二进制的指数表达法(问题2)为 1.0001 * 2^1 因为小数点 移了 1 位所以是 2^1

5. 补位 127 (问题31+127=128  128 二进制为 100000001 是 2^1 中的

6. 整体的二进制表示就是 0 10000000 00010000000000000000000  

其中数符 0 代表是正数,对应 1 代表负数, 阶码 10000000 是第五步产出的, 尾数 00010000000000000000000 是第四步的 1.0001 去除整数 1(问题4) 后补位到 23 位产出的

### 0.001 的内存二进制表达

1. 整数位 0 获取二进制为 0

2. 0.001 获取二进制为 00000000010000011000100100110111111111111111...

3. 合并就是 0.00000000010000011000100100110111111111111111...

4. 二进制的指数和表达法为 1.0000011000100100110111111111111111... * 2^-10 因为向 移动了 10 位所以是 2^-10

5. 补位 -10+127=117 二进制为=1110101, 注意这里是 7 位,下面会补 0

6. 整体的二进制表示就是 0 01110101 0000011000100100110111111111111111

内存结构同 2.125 分析过程

### 个人理解

1. 为啥这样存有什么原因,有什么好处和坏处

- 首先假如我们不这样存, 2.125 我们分成整数位和小数位分别存到内存中,那么一个整数位大小就能占用 32 位,小数位占用 32 位,总共一个 float 就占用 64 位

- 其次 0.001 这种用二进制表示如何表示,只能 00000000010000011000100100110111111111111111...,这样用 float 依然会产生精度缺失的问题, 还有就是假如 0000000......000001 这种情况怎么办,也就是前面的 0 就占了很多位,后面的数字根本存不了多大

- 好处就是上面的坏处, 占用内存小,精度依然缺失,可以存储这种很长 0 的数

2. 为啥去除指数表达式中的 1 (1.xxxxx)去除 1 留下 xxxxx,首先我们大致知道了二进制指数表达法,其本质就是为了去除小数部分的 0 开头的数据,因为 0 全部保留那么就像解答 1 中所说的那样 0 怎么去存储,这里的二进制指数表达式就很优雅的去除了 0,用的就是 2^x 来代表 0,然后把 1.xxx 中的 1 去除,因为一旦使用了二进制指数表达式,你的最前面一定是个 1,这也是这个 1 为啥可以去除
,这样 x 就可以代表偏移去除的 0 值,就可以单独代表很多小数

3. 上面 2 个例子的补位是啥操作 +127 
目前不知道

  1. 忽略 1 是因为前面都是 1 所以可以少存一位

### 总结

总的来讲,你要体会为啥要使用二进制的指数表达法,首先是为了去除整数和小数之间的区分,整体看成小数,然后用阶码来表示是变大了还是变小了,用尾数代表真正的数,然后要体会为啥是这样存储,本质就是为了存储二进制指数表达法

关联博客阅读: 

1. https://www.cnblogs.com/backwords/p/9826773.html 

2. https://blog.csdn.net/fwb330198372/article/details/70238982   

工具:

1. https://tool.lu/hexconvert/ 二进制转 10 进制

2. http://www.binaryconvert.com/result_float.html?decimal=048046048048049 浮点转二进制

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • 面试

    面试造航母,上班拧螺丝。多面试,少加班。

    324 引用 • 1395 回帖 • 3 关注
  • LeetCode

    LeetCode(力扣)是一个全球极客挚爱的高质量技术成长平台,想要学习和提升专业能力从这里开始,充足技术干货等你来啃,轻松拿下 Dream Offer!

    209 引用 • 72 回帖
  • TGIF

    Thank God It's Friday! 感谢老天,总算到星期五啦!

    284 引用 • 4481 回帖 • 656 关注
  • MongoDB

    MongoDB(来自于英文单词“Humongous”,中文含义为“庞大”)是一个基于分布式文件存储的数据库,由 C++ 语言编写。旨在为应用提供可扩展的高性能数据存储解决方案。MongoDB 是一个介于关系数据库和非关系数据库之间的产品,是非关系数据库当中功能最丰富,最像关系数据库的。它支持的数据结构非常松散,是类似 JSON 的 BSON 格式,因此可以存储比较复杂的数据类型。

    90 引用 • 59 回帖 • 2 关注
  • Kotlin

    Kotlin 是一种在 Java 虚拟机上运行的静态类型编程语言,由 JetBrains 设计开发并开源。Kotlin 可以编译成 Java 字节码,也可以编译成 JavaScript,方便在没有 JVM 的设备上运行。在 Google I/O 2017 中,Google 宣布 Kotlin 成为 Android 官方开发语言。

    19 引用 • 33 回帖 • 25 关注
  • Wide

    Wide 是一款基于 Web 的 Go 语言 IDE。通过浏览器就可以进行 Go 开发,并有代码自动完成、查看表达式、编译反馈、Lint、实时结果输出等功能。

    欢迎访问我们运维的实例: https://wide.b3log.org

    30 引用 • 218 回帖 • 604 关注
  • wolai

    我来 wolai:不仅仅是未来的云端笔记!

    1 引用 • 11 回帖 • 1 关注
  • Windows

    Microsoft Windows 是美国微软公司研发的一套操作系统,它问世于 1985 年,起初仅仅是 Microsoft-DOS 模拟环境,后续的系统版本由于微软不断的更新升级,不但易用,也慢慢的成为家家户户人们最喜爱的操作系统。

    215 引用 • 462 回帖
  • SOHO

    为成为自由职业者在家办公而努力吧!

    7 引用 • 55 回帖 • 97 关注
  • 阿里云

    阿里云是阿里巴巴集团旗下公司,是全球领先的云计算及人工智能科技公司。提供云服务器、云数据库、云安全等云计算服务,以及大数据、人工智能服务、精准定制基于场景的行业解决方案。

    89 引用 • 345 回帖 • 2 关注
  • 百度

    百度(Nasdaq:BIDU)是全球最大的中文搜索引擎、最大的中文网站。2000 年 1 月由李彦宏创立于北京中关村,致力于向人们提供“简单,可依赖”的信息获取方式。“百度”二字源于中国宋朝词人辛弃疾的《青玉案·元夕》词句“众里寻他千百度”,象征着百度对中文信息检索技术的执著追求。

    63 引用 • 785 回帖 • 252 关注
  • 微信

    腾讯公司 2011 年 1 月 21 日推出的一款手机通讯软件。用户可以通过摇一摇、搜索号码、扫描二维码等添加好友和关注公众平台,同时可以将自己看到的精彩内容分享到微信朋友圈。

    129 引用 • 793 回帖
  • SendCloud

    SendCloud 由搜狐武汉研发中心孵化的项目,是致力于为开发者提供高质量的触发邮件服务的云端邮件发送平台,为开发者提供便利的 API 接口来调用服务,让邮件准确迅速到达用户收件箱并获得强大的追踪数据。

    2 引用 • 8 回帖 • 437 关注
  • Ant-Design

    Ant Design 是服务于企业级产品的设计体系,基于确定和自然的设计价值观上的模块化解决方案,让设计者和开发者专注于更好的用户体验。

    17 引用 • 23 回帖
  • Jenkins

    Jenkins 是一套开源的持续集成工具。它提供了非常丰富的插件,让构建、部署、自动化集成项目变得简单易用。

    51 引用 • 37 回帖
  • App

    App(应用程序,Application 的缩写)一般指手机软件。

    90 引用 • 383 回帖 • 1 关注
  • AngularJS

    AngularJS 诞生于 2009 年,由 Misko Hevery 等人创建,后为 Google 所收购。是一款优秀的前端 JS 框架,已经被用于 Google 的多款产品当中。AngularJS 有着诸多特性,最为核心的是:MVC、模块化、自动化双向数据绑定、语义化标签、依赖注入等。2.0 版本后已经改名为 Angular。

    12 引用 • 50 回帖 • 422 关注
  • 设计模式

    设计模式(Design pattern)代表了最佳的实践,通常被有经验的面向对象的软件开发人员所采用。设计模式是软件开发人员在软件开发过程中面临的一般问题的解决方案。这些解决方案是众多软件开发人员经过相当长的一段时间的试验和错误总结出来的。

    198 引用 • 120 回帖
  • Telegram

    Telegram 是一个非盈利性、基于云端的即时消息服务。它提供了支持各大操作系统平台的开源的客户端,也提供了很多强大的 APIs 给开发者创建自己的客户端和机器人。

    5 引用 • 35 回帖
  • Maven

    Maven 是基于项目对象模型(POM)、通过一小段描述信息来管理项目的构建、报告和文档的软件项目管理工具。

    185 引用 • 318 回帖 • 348 关注
  • OkHttp

    OkHttp 是一款 HTTP & HTTP/2 客户端库,专为 Android 和 Java 应用打造。

    16 引用 • 6 回帖 • 54 关注
  • Lute

    Lute 是一款结构化的 Markdown 引擎,支持 Go 和 JavaScript。

    25 引用 • 191 回帖 • 19 关注
  • 周末

    星期六到星期天晚,实行五天工作制后,指每周的最后两天。再过几年可能就是三天了。

    14 引用 • 297 回帖
  • Markdown

    Markdown 是一种轻量级标记语言,用户可使用纯文本编辑器来排版文档,最终通过 Markdown 引擎将文档转换为所需格式(比如 HTML、PDF 等)。

    163 引用 • 1450 回帖 • 2 关注
  • GitLab

    GitLab 是利用 Ruby 一个开源的版本管理系统,实现一个自托管的 Git 项目仓库,可通过 Web 界面操作公开或私有项目。

    46 引用 • 72 回帖
  • 七牛云

    七牛云是国内领先的企业级公有云服务商,致力于打造以数据为核心的场景化 PaaS 服务。围绕富媒体场景,七牛先后推出了对象存储,融合 CDN 加速,数据通用处理,内容反垃圾服务,以及直播云服务等。

    25 引用 • 215 回帖 • 163 关注
  • 倾城之链
    23 引用 • 66 回帖 • 96 关注