当前位置: 首页 > news >正文

kt 中商品的金额字段使用double 还是 bigdecimal

在Kotlin中处理商品金额等金融计算时,必须使用 BigDecimal 而非 Double。以下是核心原因及实践建议:


️ 一、为什么必须用 BigDecimal

  1. Double 存在精度丢失问题
    Double 是二进制浮点数(遵循 IEEE 754 标准),无法精确表示大多数十进制小数(如 0.1)。
    示例

    println(0.1 + 0.2) // 输出 0.30000000000000004(非精确的 0.3)
    

    这种误差在金额计算中会导致累计错误,尤其涉及乘法、除法时更明显。

  2. BigDecimal 的精确性优势

    • 基于十进制存储,可精确表示任意位数的小数(如 0.1)。
    • 提供对舍入模式(RoundingMode)和精度(scale)的完全控制,确保计算结果符合金融规范。

✅ 二、Kotlin 中 BigDecimal 的最佳实践

1. 初始化:必须通过字符串构造
// ✅ 正确做法(避免二进制误差)
val price = BigDecimal("19.99")//  ❌ 错误做法(直接传 Double 仍会引入误差)
val errorPrice = BigDecimal(19.99) // 实际值可能是 19.990000000000002
2. 运算:显式指定精度和舍入模式
val price = BigDecimal("100.55")
val taxRate = BigDecimal("0.07")// 乘法 + 保留两位小数(四舍五入)
val tax = price.multiply(taxRate).setScale(2, RoundingMode.HALF_UP) // 除法:必须指定精度,否则抛 ArithmeticException
val divided = price.divide(BigDecimal(3), 2, RoundingMode.HALF_UP)
3. 比较数值:用 compareTo() 而非 equals()
val a = BigDecimal("1.0")
val b = BigDecimal("1.00")println(a == b) // false(equals() 严格比较小数位数)
println(a.compareTo(b) == 0) // true(比较实际数值)

三、Double 的适用场景(非金额)

仅适用于科学计算、图形渲染等容忍误差的场景:

// 物理引擎计算(精度要求低)
val velocity = 3.0 * 9.8 

🔧 四、封装工具类(推荐)

import java.math.BigDecimal
import java.math.RoundingModeobject MoneyUtils {private const val DEFAULT_SCALE = 2// 安全转换 Double → BigDecimal(字符串中转)fun fromDouble(value: Double): BigDecimal {return BigDecimal(value.toString())}// 金额标准化(保留两位小数)fun normalize(amount: BigDecimal): BigDecimal {return amount.setScale(DEFAULT_SCALE, RoundingMode.HALF_UP)}// 安全除法fun divide(a: BigDecimal, b: BigDecimal, scale: Int = DEFAULT_SCALE): BigDecimal {return a.divide(b, scale, RoundingMode.HALF_UP)}
}// 使用示例
val total = MoneyUtils.fromDouble(100.0).multiply(BigDecimal("0.85")).let { MoneyUtils.normalize(it) }

💾 五、数据库存储方案

方案数据类型Java/Kotlin 类型优点缺点
推荐 ✅DECIMAL(18,2)BigDecimal精确存储,代码直观占用空间略大
高性能场景(分单位)BIGINTLong存储小,查询快需手动转换元/分单位

📌 总结

  • 金额字段必须用 BigDecimal:避免 Double 的二进制精度陷阱。
  • 通过字符串初始化BigDecimal("123.45") 是唯一安全方式。
  • 显式控制精度和舍入:所有运算(尤其除法)需指定 scaleRoundingMode
  • 数据库存储匹配:使用 DECIMAL(p,s) 类型直接映射 BigDecimal

在金融系统中,精度错误可能导致资损或法律纠纷,务必严格遵循上述规范。

http://www.dtcms.com/a/279850.html

相关文章:

  • 动态规划题解——最长递增子序列【LeetCode】记忆化搜索方法
  • 【每日刷题】杨辉三角
  • Git根据标签Tag强制回滚版本
  • 面试常问:如何在一个长度为n的无序数据快速获取前k个数值
  • 网络传输过程
  • GaussDB between的用法
  • 光伏板如何最大化铺设?
  • 【PostgreSQL异常解决】`PostgreSQL`异常之类型转换错误
  • 记录自己在将python文件变成可访问库文件是碰到的问题
  • vert.x 官网docs, vert.x中文文档地址 vertx文档
  • 文心4.5开源之路:引领技术开放新时代!
  • 【前端:Typst】--let关键字的用法
  • 高德开放平台携手阿里云,面向开发者推出地图服务产品MCP Server
  • 外部协作不力影响项目进度,如何加强外部沟通
  • 项目进度压缩影响质量,如何平衡进度与质量
  • LeetCode|Day11|557. 反转字符串中的单词 III|Python刷题笔记
  • 稀土化合物:助力高效种植与健康养殖
  • vue笔记3 VueRouter VueX详细讲解
  • 对象的使用
  • CAN终端电阻为什么是60R+60R,而不直接用120R?
  • 前端vue对接海康摄像头流程
  • Flink窗口处理函数
  • C++-linux 5.gdb调试工具
  • 【从语言幻觉看趋势】从语言幻觉到多智能体协作:GPT多角色系统的技术演进与实践路径
  • 判断端口处于监听状态的方法
  • 腾讯云WAF域名分级防护实战笔记
  • EPLAN 电气制图(八):宏应用与变频器控制回路绘制全攻略
  • ssm学习笔记day07mybatis
  • 如何在 Shopify 中创建退货标签
  • 【C语言】浮点数在内存中的存储:从科学计数法到内存存储