0.1 + 0.2 不等于 0.3

在前端(JavaScript)里 0.1 + 0.2 ≠ 0.3,核心原因是 十进制小数在二进制中无法被精准表示,导致运算时出现精度丢失------这不是JS的bug,而是所有使用「二进制浮点数」的编程语言(如Python、Java)都存在的通用问题。

一、核心原理:十进制→二进制的精度丢失

我们使用用十进制 (0-9),但计算机底层只能用二进制 (0和1)。像 0.10.2 这样的十进制小数,转换成二进制后是无限循环小数 ,就像十进制中 1/3 = 0.333333... 无法写尽一样。

1. 具体转换过程(简化版)
  • 0.1 的二进制:0.0001100110011...0011 无限循环)

  • 0.2 的二进制:0.001100110011...0011 无限循环)

但JS使用的是 64位 双精度浮点数 标准(IEEE 754),只能存储有限位数(约16位有效数字),所以会对无限循环的二进制数做「舍入截断」,最终存储的 0.10.2 都是近似值,而非精确值:

  • 实际存储的 0.10.10000000000000000555

  • 实际存储的 0.20.20000000000000001110

2. 相加后的结果

这两个近似值相加:

0.10000000000000000555 + 0.20000000000000001110 = 0.30000000000000001665

所以我们看到的 0.1 + 0.2 结果是 0.30000000000000004(不同环境舍入略有差异),而非精确的 0.3

二、代码验证:直观看到精度问题

javascript 复制代码
// 验证存储的近似值
console.log(0.1); // 输出 0.1(浏览器会美化显示,但实际是近似值)
console.log(0.1.toPrecision(20)); // 输出 0.10000000000000000555(真实存储值)
console.log(0.2.toPrecision(20)); // 输出 0.20000000000000001110

// 核心运算
console.log(0.1 + 0.2); // 输出 0.30000000000000004(≠ 0.3)
console.log(0.1 + 0.2 === 0.3); // 输出 false(重点!判断相等会出错)

三、前端开发中如何解决这个问题?

核心思路是 避免直接比较浮点数,或把小数转为整数运算后再转回小数,推荐3种常用方法:

方法1:固定小数位数(toFixed,适合简单场景)
javascript 复制代码
// toFixed(n) 保留n位小数(返回字符串,需转数字)
const result = (0.1 + 0.2).toFixed(1); 
console.log(result); // 输出 "0.3"
console.log(Number(result) === 0.3); // 输出 true

// 注意:toFixed会四舍五入,适合金额、简单计算
console.log((0.15).toFixed(1)); // 输出 "0.2"(四舍五入)
方法2:转整数运算(精准,适合金额/常用计算)

把小数乘以10的n次方转为整数(消除小数),运算后再除以10的n次方:

javascript 复制代码
// 0.1+0.2 → 转成 1+2=3 → 再÷10=0.3
function addFloat(a, b) {
  // 取最大小数位数(0.1是1位,0.2是1位,max=1)
  const decimalLen = Math.max(
    a.toString().split(".")[1]?.length || 0,
    b.toString().split(".")[1]?.length || 0
  );
  const multiplier = 10 ** decimalLen; // 10^1=10
  // 转整数运算,避免精度丢失
  return (a * multiplier + b * multiplier) / multiplier;
}

console.log(addFloat(0.1, 0.2)); // 输出 0.3
console.log(addFloat(0.15, 0.25)); // 输出 0.4
方法3:使用专业的精度库(适合复杂计算/金融场景)

如果涉及大量浮点数运算(如电商金额、财务系统),推荐用成熟库(如 decimal.jsbig.js):

bash 复制代码
# 安装
npm install decimal.js
javascript 复制代码
import Decimal from 'decimal.js';

const a = new Decimal(0.1);
const b = new Decimal(0.2);
const result = a.plus(b).toNumber(); // plus是decimal.js的加法方法

console.log(result); // 输出 0.3
console.log(result === 0.3); // 输出 true

总结

  1. 0.1 + 0.2 ≠ 0.3 的本质是 十进制小数转二进制时的无限循环,导致浮点数存储为近似值,相加后精度丢失;

  2. 这是IEEE 754二进制浮点数的通用特性,并非JavaScript的bug;

  3. 前端解决思路:简单场景用 toFixed,通用场景转整数运算,复杂/金融场景用 decimal.js 等专业库,永远不要直接用 === 判断浮点数是否相等

相关推荐
椰猫子19 小时前
html、css入门
开发语言·javascript·ecmascript
SuniaWang19 小时前
《Spring AI + 大模型全栈实战》学习手册系列·专题一:《RAG技术全景解析:从原理到架构设计》
java·javascript·人工智能·spring boot·后端·spring·架构
Irene199119 小时前
通过JavaScript创建新的img元素并指定onload事件,为什么要在赋值src属性之前指定事件处理程序?
javascript
tobias.b20 小时前
计算机基础知识-操作系统
考研·面试·职场和发展
爱丽_20 小时前
Vue3 响应式系统:`ref`/`reactive`/`watchEffect` 的工作方式与最佳实践
前端·vue.js
~无忧花开~20 小时前
React元素渲染:核心概念全解析
开发语言·前端·javascript·react.js
harrain20 小时前
antv x6graph使用经验
前端·antv·x6
开开心心就好20 小时前
免费无广告的礼金记账本,安卓应用
java·前端·ubuntu·edge·pdf·负载均衡·语音识别
qq_2837200520 小时前
nestjs实战(六):诺依Nest.js + MySQL 项目改造为兼容达梦8数据库详细教程
javascript·数据库·mysql·达梦·nest.js·诺依
向往着的青绿色20 小时前
完全平方数【Letcode279题解】
开发语言·c++·数学·算法·面试·性能优化·动态规划