JavaScript 中 0.1 + 0.2 != 0.3:浮点数运算的陷阱

2024-01-11 22:44

本文主要是介绍JavaScript 中 0.1 + 0.2 != 0.3:浮点数运算的陷阱,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

引言

在日常的 JavaScript 开发中,我们经常会面对一些看似简单的数学运算。然而,当我们执行 0.1 + 0.2 这样的操作时,很多人可能会感到困惑,因为预期的结果应该是 0.3,但实际上却是一个略微不同的值。这个现象是由于 JavaScript 中浮点数运算的特性导致的,本文将深入探讨这个问题的背后原因以及如何在开发中避免相关的陷阱。

为什么会发生这种情况?

在计算机中,浮点数是用二进制表示的,而不是十进制。由于这一事实,一些十进制小数可能无法精确地用二进制表示,从而导致在浮点数运算中产生舍入误差。JavaScript 使用的是 IEEE 754 标准来表示浮点数,这一标准采用二进制表示法,因此在某些情况下,十进制小数无法完全精确映射到二进制表示。

在具体到 0.1 + 0.2 的例子中,这两个数字在二进制中的表示并不是精确的。虽然它们在十进制下是简单的小数,但在二进制中,它们的精确表示是一个无限循环小数。由于计算机的存储和运算能力的限制,这样的无限循环小数无法被精确表示,因此就会引发舍入误差。

console.log(0.1 + 0.2); // 输出 0.30000000000000004

浮点数运算的挑战

1. 精度丢失

在 JavaScript 中,由于浮点数运算的不可避免的舍入误差,可能会导致精度丢失。这意味着在比较两个浮点数是否相等时,应该使用一些技巧,而不是直接使用相等运算符。

console.log(0.1 + 0.2 === 0.3); // 输出 false
console.log(Math.abs(0.1 + 0.2 - 0.3) < Number.EPSILON); // 输出 true

2. 运算顺序的影响

JavaScript 的浮点数运算还受到运算顺序的影响。由于浮点数的表示范围和精度有限,当进行多个浮点数的运算时,不同的运算顺序可能导致不同的结果。

console.log(0.1 + 0.2 + 0.3); // 输出 0.6000000000000001
console.log(0.3 + 0.2 + 0.1); // 输出 0.6

3. 解决方案:使用整数进行运算

为了避免浮点数运算的陷阱,一种常见的做法是将浮点数转换为整数进行运算,然后再将结果转换回浮点数。这可以通过乘以一个足够大的倍数来实现。

let result = (0.1 * 10 + 0.2 * 10) / 10;
console.log(result); // 输出 0.3

如何在开发中避免问题?

在实际的开发中,我们可以采取一些策略来最小化浮点数运算带来的问题。

1. 使用整数进行运算

如前所述,将浮点数转换为整数进行运算是一种可行的解决方案。通过乘以一个足够大的倍数,我们可以避免舍入误差。

2. 使用专门的库

一些专门处理精确数学运算的库,如 Decimal.js 和 big.js,提供了更精确的十进制运算,可以在避免浮点数问题的同时保持较高的性能。

3. 使用 Number.EPSILON 进行比较

在比较浮点数是否相等时,应该使用一个很小的常量,例如 Number.EPSILON,而不是直接使用相等运算符。

console.log(Math.abs(0.1 + 0.2 - 0.3) < Number.EPSILON); // 输出 true

4. 注意运算顺序

在进行多个浮点数的运算时,应该注意运算顺序可能带来的不同结果。在关键的业务逻辑中,可以考虑重新排列运算的顺序,以最小化舍入误差的影响。

5.自定义向上向下取整

只要跟钱有关,那这个问题就必须得重视了,js中并没有针对精度保留向上向下的方法,当时我们可以利用原生的 Math.powMath.floor 二次封装,根据实际场景决定向上保留精度还是向下保留精度;

/*** 向下取整 例如:3.33333333 取3位,返回 3.333* num 原始值* precision 保留精度*/
Math.floorPow = function(num, precision){// 根据保留精度放大 小数位let magnify =  Math.pow(10, precision);// 返回小于等于给定数字的 最大整数return Math.floor(num * magnify)/magnify;
}/*** 向上取整 例如:3.33333333 取3位,返回 3.334* num 原始值* precision 保留精度*/
Math.ceilPow = function(num, precision){// 根据保留精度放大 小数位let magnify =  Math.pow(10, precision);// 返回小于等于给定数字的 最大整数return Math.ceil(num * magnify)/magnify;
}

结语

JavaScript 中 0.1 + 0.2 不等于 0.3 的现象并非 bug,而是由于浮点数运算的本质导致的。了解浮点数运算的特性,采用适当的解决方案,可以帮助我们避免在开发中因为这类问题而浪费时间。通过合理的数学运算和注意运算顺序,我们可以更好地利用 JavaScript 的数值计算功能,确保我们的应用在处理数学运算时保持准确性和可靠性。

这篇关于JavaScript 中 0.1 + 0.2 != 0.3:浮点数运算的陷阱的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/595937

相关文章

Spring AI 实现 STDIO和SSE MCP Server的过程详解

《SpringAI实现STDIO和SSEMCPServer的过程详解》STDIO方式是基于进程间通信,MCPClient和MCPServer运行在同一主机,主要用于本地集成、命令行工具等场景... 目录Spring AI 实现 STDIO和SSE MCP Server1.新建Spring Boot项目2.a

spring security 超详细使用教程及如何接入springboot、前后端分离

《springsecurity超详细使用教程及如何接入springboot、前后端分离》SpringSecurity是一个强大且可扩展的框架,用于保护Java应用程序,尤其是基于Spring的应用... 目录1、准备工作1.1 引入依赖1.2 用户认证的配置1.3 基本的配置1.4 常用配置2、加密1. 密

Spring Boot 集成 Solr 的详细示例

《SpringBoot集成Solr的详细示例》:本文主要介绍SpringBoot集成Solr的详细示例,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友一起看看吧... 目录环境准备添加依赖配置 Solr 连接定义实体类编写 Repository 接口创建 Service 与 Controller示例运行

Spring Cloud GateWay搭建全过程

《SpringCloudGateWay搭建全过程》:本文主要介绍SpringCloudGateWay搭建全过程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐... 目录Spring Cloud GateWay搭建1.搭建注册中心1.1添加依赖1.2 配置文件及启动类1.3 测

Java如何将文件内容转换为MD5哈希值

《Java如何将文件内容转换为MD5哈希值》:本文主要介绍Java如何将文件内容转换为MD5哈希值的实现方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录Java文件内容转换为MD5哈希值一个完整的Java示例代码代码解释注意事项总结Java文件内容转换为MD5

Spring Boot拦截器Interceptor与过滤器Filter深度解析(区别、实现与实战指南)

《SpringBoot拦截器Interceptor与过滤器Filter深度解析(区别、实现与实战指南)》:本文主要介绍SpringBoot拦截器Interceptor与过滤器Filter深度解析... 目录Spring Boot拦截器(Interceptor)与过滤器(Filter)深度解析:区别、实现与实

解决Java异常报错:java.nio.channels.UnresolvedAddressException问题

《解决Java异常报错:java.nio.channels.UnresolvedAddressException问题》:本文主要介绍解决Java异常报错:java.nio.channels.Unr... 目录异常含义可能出现的场景1. 错误的 IP 地址格式2. DNS 解析失败3. 未初始化的地址对象解决

SpringBoot后端实现小程序微信登录功能实现

《SpringBoot后端实现小程序微信登录功能实现》微信小程序登录是开发者通过微信提供的身份验证机制,获取用户唯一标识(openid)和会话密钥(session_key)的过程,这篇文章给大家介绍S... 目录SpringBoot实现微信小程序登录简介SpringBoot后端实现微信登录SpringBoo

Java中的StringUtils.isBlank()方法解读

《Java中的StringUtils.isBlank()方法解读》:本文主要介绍Java中的StringUtils.isBlank()方法,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑... 目录所在库及依赖引入方法签名方法功能示例代码代码解释与其他方法的对比总结StringUtils.isBl

如何合理使用Spring的事务方式

《如何合理使用Spring的事务方式》:本文主要介绍如何合理使用Spring的事务方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、介绍1.1、底层构造1.1.事务管理器1.2.事务定义信息1.3.事务状态1.4.联系1.2、特点1.3、原理2. Sprin