1 前言

来源:公众号(c语言与cpp编程)

我们在学习 C 语言时,通常认为浮点数和小数是等价的,并没有严格区分它们的概念,这也并没有影响到我们的学习,原因就是浮点数和小数是绑定在一起的,只有小数才使用浮点格式来存储。

其实,整数和小数可以都使用定点格式来存储,也可以都使用浮点格式来存储,但实际情况却是,C 语言使用定点格式存储整数,使用浮点格式存储小数,这是在 “数值范围” 和 “数值精度” 两项重要指标之间追求平衡的结果。

2 什么是浮点数?

浮点型简单讲就是实数的意思。浮点数在计算机中用以近似表示任意某个实数。具体的说,这个实数由一个整数或定点数(即尾数)乘以某个基数(计算机中通常是 2)的整数次幂得到,这种表示方法类似于基数为 10 的科学记数法。

3 浮点数在内存中的存储

首先明确一点,无论是整型、浮点型还是字符等等数据类型在计算机底层都是以二进制的方式存储的。

浮点数在内存中的存储和整数不同,因为整数都可以转换为一一对应的二进制数据。而浮点数的存储是由符号位 (sign) + 指数位 (exponent) + 小数位 (fraction) 组成。

类型 符号位 指数 尾数 Float 1位(第31位) 8位(第23~30位) 23位(第0~22位) Double 1位(第63位) 11位(第52~62位) 52位(第0~51位)

int 和 float 同样占据四个字节的内存,但是 float 所能表示的最大值比 int 大得多,其根本原因是浮点数在内存中是以指数的方式存储。

浮点数转换到内存中存储的步骤分为如下三步:

  • 将浮点数转换成二进制
  • 用科学计数法表示二进制浮点数
  • 计算指数偏移后的值

对于第3点:计算指数时需要加上偏移量(后面有介绍为什么使用偏移量),而偏移量的值与浮点数的类型有关( float 偏移量值为 127 ,double 偏移量值为 1023)。比方对于指数 6,float 与 double 类型偏移后的值分别为:

  • float : 127 + 6 = 133
  • double:1023 + 6 = 1029

4 实例

浮点数19.625用float是如何存储的:

  • 将浮点数转换成二进制:10011.101(将 19.625 整数部分采用除 2 取余,小数部分采用乘 2 取整法);
  • 用科学计数法表示二进制浮点数:1.0011101*2^4;
  • 计算指数偏移后的值:127 + 4 = 131 (10000011);
  • 拼接综上所述,float 类型的 19.625 在内存中的值为:0 - 10000011 - 001 1101 0000 0000 0000 0000。

5 float与double范围和精度

范围

float和double的范围是由指数的位数来决定的。(因为表示的时候都是1.x * 2^Y的形式,所以忽略了1.x的效果,直接取指数表示浮点数的范围)

  • float:

1bit(符号位) 8bits(指数位) 23bits(尾数位)

  • double:

1bit(符号位) 11bits(指数位) 52bits(尾数位)

于是,float的指数范围为-127~+128,而double的指数范围为-1023~+1024,并且指数位是按补码的形式来划分的。

其中负指数决定了浮点数所能表达的绝对值最小的非零数;而正指数决定了浮点数所能表达的绝对值最大的数,也即决定了浮点数的取值范围。

float的范围为-2^128 ~ +2^128,也即-3.40E+38 ~ +3.40E+38;

double的范围为-2^1024 ~ +2^1024,也即-1.79E+308 ~ +1.79E+308。

精度

float和double的精度是由尾数的位数来决定的,尾数越多能表示的小数点后面有效数字就越多,因此精度就越高。浮点数在内存中是按科学计数法来存储的,其整数部分始终是一个隐含着的“1”,由于它是不变的,故不能对精度造成影响。

float:2^23 = 8388608,一共七位,这意味着最多能有 7 位有效数字,但绝对能保证的为 6 位,也即float的精度为 6~7 位有效数字;

double:2^52 = 4503599627370496,一共 16 位,同理,double的精度为 15~16 位。

6 解剖:为什么要用偏移量的方式来计算指数?

如果不采用偏移量的方式:

8 位 2 进制数表示的有符号数范围有两个区间:0000 0000~0111 1111和1000 0000~1111 1111,分别为0~+127和-127~0。

大家看到这里的问题了吧,有两个 0 ,一个正 0 和一个负 0。

如果采用偏移量的方式:

127 转化为二进制是:0111 1111

那么

  • 当我们要表示 -127,则有127-127即0111 1111 - 0111 1111 = 0000 0000
  • 当我们要表示 -126,则有127-126即0111 1111 - 0111 1110 = 0000 0001
  • 当我们要表示 -2,则有127-2即0111 1111 - 0000 0010 = 0111 1101
  • 当我们要表示 -1,则有127-1即0111 1111 - 0000 0001 = 0111 1110
  • 当我们要表示 0,则有0+127即0000 0000 + 0111 1111 = 0111 1111
  • 当我们要表示 1,则有1+127即0000 0001 + 0111 1111 = 1000 0000
  • 当我们要表示 2,则有1+127即0000 0010 + 0111 1111 = 1000 0001

当我们要表示128,则有128+127即1000 0000 + 0111 1111 = 1111 1111

由上面的例子,我们可以得出规律,采用移位存储技术,我们可以使用 8 位二进制来表示从-127~+128共计 127 个负数+零(0)+ 128 个正数总共 256 个数,看来使用移位存储既没有 +0 和 -0 的问题,又能充分使用新生成的8位二进制数最大限度的表示单精度浮点数的幂指数,是非常合理的。

c语言整数转浮点数_浮点数的秘密相关推荐

  1. 64位浮点数_浮点数运算的机器误差分析

    感觉跟专栏主题不是很搭... 不过这是这学期计算物理的作业,还是放上来吧,也算勉强沾边了吧. 用一个浮点数相加的例子来演示计算机在计算时所产生的误差. 在Python中,用0.2+0.4 会得到0.6 ...

  2. C语言整数怎么转化为浮点数,C语言浮点数和整数转换的分析

    这里以C语言的浮点数为例,在IEEE浮点标准下,整数转换为浮点数的过程做一些直观的分析和具体的实现,IEEE浮点数的标准细节详见(). 在C语言中,使用float和double类型数据分别对应单精度和 ...

  3. c# 取余数 浮点数_浮点数精度问题透析:小数计算不准确+浮点数精度丢失根源

    在知乎上上看到如下问题: 浮点数精度问题的前世今生? 1.该问题出现的原因 ? 2.为何其他编程语言,比如java中可能没有js那么明显 3.大家在项目中踩过浮点数精度的坑? 4.最后采用哪些方案规避 ...

  4. java 正则 浮点数_浮点数的正则表达式

    蝴蝶不菲 使用[.]而不是\.和[0-9]而不是\d以避免在某些语言(如Java)中逃避问题.感谢无名者一开始就认识到了这一点.一个相对简单的模式匹配浮点数是[+-]?([0-9]*[.])?[0-9 ...

  5. 64位浮点数_浮点数误差

    chopper:目录​zhuanlan.zhihu.com 本篇文章通过解析浮点数精度格式,分析为什么浮点数会出现误差,并提供了减少浮点数运算误差要尽量遵循的原则. 在计算机中,小数有两种表示方式:定 ...

  6. 单精度浮点数和双精度浮点数_浮点数和双精度数之间的区别

    单精度浮点数和双精度浮点数 Here you will learn about difference between float and double i.e. float vs double. 在这 ...

  7. c语言 整数和浮点数_C ++处理整数和浮点数

    c语言 整数和浮点数 C ++中的所有数字 ( All About Numbers in C++ ) In C++ there are two types of numbers. Ints and f ...

  8. 可以获取python整数类型帮助的是什么-Python 的数值类型(整数、长整数、浮点数和复数)...

    在Python中有4种类型的数--整数.长整数.浮点数和复数. 2是一个整数的例子. 长整数不过是大一些的整数. 3.23和52.3E-4是浮点数的例子.E标记表示10的幂.在这里,52.3E-4表示 ...

  9. python整数类型-Python 的数值类型(整数、长整数、浮点数和复数)

    在Python中有4种类型的数--整数.长整数.浮点数和复数. 2是一个整数的例子. 长整数不过是大一些的整数. 3.23和52.3E-4是浮点数的例子.E标记表示10的幂.在这里,52.3E-4表示 ...

  10. 用正则表达式求非负整数 、匹配正整数、非正整数、负整数、整数、非负浮点数、正浮点数、非正浮点数、负浮点数、浮点数、有数字26个英文字母组成的字符串

    用正则表达式求非负整数 .匹配正整数.非正整数.负整数.整数.非负浮点数.正浮点数.非正浮点数.负浮点数.浮点数.有数字26个英文字母组成的字符串 // *** // 1.求非负整数: var str ...

最新文章

  1. LInux 文件系统 tmpfs 分区不显示解决
  2. canvas百分比加载动画
  3. cannot be cast to org.springframework.web.accept.ContentNegotiationManager
  4. Spark程序运行常见错误解决方法以及优化
  5. php中如何写js代码提示_PHP 如何编写类似js中alert() 提示框
  6. DCMTK:TLS测试DcmSCP和DcmSCPPool类
  7. IDaaS企业身份管理训练营火热报名中 体验新一代企业云身份服务
  8. 200730学习日报6字符串
  9. Consul入门04 - Consul集群 1
  10. PaperWeekly 第28期 | 图像语义分割之特征整合和结构预测
  11. primefaces_懒惰的JSF Primefaces数据表分页–第1部分
  12. 51单片机4位数乘法C语言,51单片机实现4位数以内的加减法
  13. clion 使用技巧
  14. 李彦宏告诫年轻人:向前看两年
  15. 射频识别技术漫谈(25)——Felica简介
  16. 阿里云Blink介绍
  17. kaggle使用tpu
  18. 无法打开登录所请求的数据库 。登录失败。用户*登录失败。解决办法
  19. 正在更新office 请稍后... 解决办法
  20. 05,JavaScript脚本中cookie

热门文章

  1. UVALive 5135 Mining Your Own Business 双连通分量
  2. 4.Prometheus 监控技术与实践 --- 服务发现
  3. 54.Linux/Unix 系统编程手册(下) -- POSIX 共享内存
  4. 49. 模型层 --- dao 层
  5. 4. Javascript 函数
  6. java Session生命周期讨论
  7. 《数学之美》—PageRank
  8. transform-origin用于设置变换原点
  9. 程序设计中为什么要解耦?
  10. JavaWeb将图片显示在浏览器中