【Rust 课外知识】0.1+0.2为什么不等于0.3(上)

简介: 【Rust 课外知识】0.1+0.2为什么不等于0.3(上)

0x00 开篇


0.1 + 0.2 = 0.3这个等式,在数学里是没有任何问题的。但是为什么我们使用Rust求解时得到的结果却不是0.3呢?这篇文章,我们就深刻的剖析一下。


0x01 先看结果


首先,我们先用代码看下使用Rust计算0.1 + 0.2 的结果,如下图:


0a2653c851af460fa595bd959398a8f1.png


其结果是0.30000000000000004。那这是什么原因呢?其实这与计算机是如何保存数据有关系。


0x02 进制与转换


要寻求原因,这里有几个概念要先了解下。


进制


我们在生活中最常见的有10进制,逢10进1,还有60进制,1分钟等于60秒,另外还有24进制,1天等于24小时等等。然而计算机所识别的只有2进制,它只认识0和1。那么这就存在进制转换的问题。


转换


10进制如何转二进制?这里有个口诀:整数部分除以2,逆序取余;小数部分乘以2,顺序取整。下面以30.25为例,解释下。

整数部分
30 ÷ 2 = 15······0                      
15 ÷ 2 =  7······1                       
 7 ÷ 2 =  3······1
 3 ÷ 2 =  1······1
 1 ÷ 2 =  0······1
小数部分
0.25 × 2 = 0.5  =>  0
0.5 × 2 = 1.0  =>  1

按照规则,30.25的二进制表示形式为11110.01


0x03 科学计数法(指数)


[WHAT] 什么是科学计数法?


任意十进制数,都可以表示为一个指数形式,如 D = A × 10^n。同理,任意二进制数也可以表示成指数形式,如 B = A × 2^n。(n为指数)

十进制数 2022 可以表示为 : 2.022 × 10 ^ 3
二进制数 0.0111 可以表示为 : 1.11 × 2 ^ 10 (这里不是十进制的“十”,而是二进制的“二”)


[WHY] 为什么要使用指数?


使用科学计数法,可以在计算机有限的存储空间中表示更大的数据范围。其如下图指示,以一个6位的十进制整数为例,如果不使用靠科学计数,那么它最大可以表示的数字为99999。然而采用科学计数法,假设前3位表示尾数,后3位表示指数,则它最大可以表示9.999 × 10 ^ 999。它们相差多少倍,是显而易见的。


2d65d23f6d4748949b924e4057485923.png


0x04 浮点数在计算机中的表现形式


计算机中的浮点数的运算大都采用IEEE 754 的标准来运算。以64位浮点数为例,IEEE754标准规定,第1位表示符号位,第2~12位表示指数(共11位),第13~64位表示尾数(共52位)。如下图所示。


6de278e6d6694ce5bb08e7e842b7e74b.png


0x05 结语


本文主要介绍了进制,进制的转换,和二进制的指数表示形式。在计算机中的浮点数又遵循IEEE 754标准运算,其实这也是导致0.1 + 0.2 不等于 0.3的重要原因,下节将会详细来说一下IEEE 754标准的浮点数运算为什么会得出这个结果。

相关文章
|
Rust Java 调度
【Rust 课外知识】Rust中的三种多态性(下)——Enum + Struct
【Rust 课外知识】Rust中的三种多态性(下)——Enum + Struct
【Rust 课外知识】Rust中的三种多态性(下)——Enum + Struct
|
Rust C++
【Rust 课外知识】你还不知道的Rust10个小技巧(下)
【Rust 课外知识】你还不知道的Rust10个小技巧(下)
【Rust 课外知识】你还不知道的Rust10个小技巧(下)
|
Rust Unix Go
【Rust 课外知识】你还不知道的Rust10个小技巧(上)
【Rust 课外知识】你还不知道的Rust10个小技巧(上)
【Rust 课外知识】你还不知道的Rust10个小技巧(上)
|
Rust JavaScript 前端开发
【Rust 课外知识】0.1+0.2为什么不等于0.3(下)
【Rust 课外知识】0.1+0.2为什么不等于0.3(下)
【Rust 课外知识】0.1+0.2为什么不等于0.3(下)
|
缓存 Rust Java
【Rust 课外知识】Rust中的三种多态性(中)——Trait的两种方式
【Rust 课外知识】Rust中的三种多态性(中)——Trait的两种方式
|
存储 Rust Java
【Rust 课外知识】Rust中的三种多态性(上)——Enum和Trait
【Rust 课外知识】Rust中的三种多态性(上)——Enum和Trait
|
7天前
|
Rust 安全 Java
探索Rust语言的并发编程模型
探索Rust语言的并发编程模型
|
14天前
|
Rust 安全 区块链
探索Rust语言:系统编程的新选择
【10月更文挑战第27天】Rust语言以其安全性、性能和并发性在系统编程领域受到广泛关注。本文介绍了Rust的核心特性,如内存安全、高性能和强大的并发模型,以及开发技巧和实用工具,展示了Rust如何改变系统编程的面貌,并展望了其在WebAssembly、区块链和嵌入式系统等领域的未来应用。
|
15天前
|
Rust 安全 Java
编程语言新宠:Rust语言的特性、优势与实战入门
【10月更文挑战第27天】Rust语言以其独特的特性和优势在编程领域迅速崛起。本文介绍Rust的核心特性,如所有权系统和强大的并发处理能力,以及其性能和安全性优势。通过实战示例,如“Hello, World!”和线程编程,帮助读者快速入门Rust。
32 1
|
16天前
|
Rust 安全 编译器
编程语言新宠:Rust语言的特性、优势与实战入门
【10月更文挑战第26天】Rust语言诞生于2006年,由Mozilla公司的Graydon Hoare发起。作为一门系统编程语言,Rust专注于安全和高性能。通过所有权系统和生命周期管理,Rust在编译期就能消除内存泄漏等问题,适用于操作系统、嵌入式系统等高可靠性场景。
28 2