【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
|
12天前
|
Rust 安全 Go
揭秘Rust语言:为何它能让你在编程江湖中,既安全驰骋又高效超车,颠覆你的编程世界观!
【8月更文挑战第31天】Rust 是一门新兴的系统级编程语言,以其卓越的安全性、高性能和强大的并发能力著称。它通过独特的所有权和借用检查机制解决了内存安全问题,使开发者既能享受 C/C++ 的性能,又能避免常见的内存错误。Rust 支持零成本抽象,确保高级抽象不牺牲性能,同时提供模块化和并发编程支持,适用于系统应用、嵌入式设备及网络服务等多种场景。从简单的 “Hello World” 程序到复杂的系统开发,Rust 正逐渐成为现代软件开发的热门选择。
29 1
|
24天前
|
Rust 安全 编译器
初探 Rust 语言与环境搭建
Rust 是一门始于2006年的系统编程语言,由Mozilla研究员Graydon Hoare发起,旨在确保内存安全而不牺牲性能。通过所有权、借用和生命周期机制,Rust避免了空指针和数据竞争等问题,简化了并发编程。相较于C/C++,Rust在编译时预防内存错误,提供类似C++的语法和更高的安全性。Rust适用于系统编程、WebAssembly、嵌入式系统和工具开发等领域。其生态系统包括Cargo包管理器和活跃社区。学习资源如"The Book"和"Rust by Example"帮助新手入门。安装Rust可通过Rustup进行,支持跨平台操作。
初探 Rust 语言与环境搭建
|
12天前
|
Rust 安全 程序员
Rust 语言的防错机制太惊人了!安全编码从此不再是难题,快来一探究竟!
【8月更文挑战第31天】《安全编码原则:Rust 语言中的防错机制》探讨了代码安全的重要性,并详细介绍了Rust语言如何通过内存安全模型、所有权与借用规则等特性,在编译阶段检测并阻止潜在错误,如缓冲区溢出和悬空指针。文章还讨论了类型安全、边界检查等其他安全特性,并提出了遵循不可变引用、避免裸指针及充分测试等实用编码原则,以进一步提升代码质量和安全性。随着Rust在软件开发中的应用日益广泛,掌握其安全编码原则变得尤为重要。
26 0
|
12天前
|
Rust 安全 调度
从零构建梦想操作系统:用Rust语言亲手打造专属内核,你也可以成为系统开发者!
【8月更文挑战第31天】开发操作系统内核虽具挑战,却也充满乐趣。本文将指导你从零开始,使用Rust语言构建一个简单的操作系统内核。首先安装Rust环境和交叉编译工具链,然后创建新项目`my_kernel`。通过修改`Cargo.toml`和编写启动函数,结合串口输出和`multiboot2`库,最终使用QEMU运行内核。此教程旨在帮助你理解Rust在系统开发中的应用,激发深入探索的兴趣。
31 1