
本文介绍两种精确截断 double 值小数点后两位(向下截断,非四舍五入)的可靠方法:使用 decimalformat 配合 roundingmode.down,或通过 bigdecimal 的 setscale() 实现无损精度控制。
在 Java 中,printf("%1.2f", x) 或 String.format() 默认采用四舍五入(HALF_UP),无法满足“严格截断”需求——例如 19.867 应变为 19.86 而非 19.87。要实现向下截断(truncation toward zero),必须显式指定舍入模式。以下是两种推荐方案:
✅ 方法一:使用 DecimalFormat + RoundingMode.DOWN
import java.text.DecimalFormat;
import java.math.RoundingMode;
double number1 = 19.867;
DecimalFormat df = new DecimalFormat("#.##");
df.setRoundingMode(RoundingMode.DOWN);
System.out.println(df.format(number1)); // 输出:19.86⚠️ 注意:"#.##" 模式会自动省略末尾零(如 19.80 → "19.8")。若需强制保留两位小数(如 "19.80"),应改用 "0.00" 模式:
df.applyPattern("0.00"); // 确保输出恒为两位小数✅ 方法二:使用 BigDecimal(更精准、推荐用于金融计算)
import java.math.BigDecimal;
import java.math.RoundingMode;
double number1 = 19.867;
BigDecimal bd = new BigDecimal(Double.toString(number1)) // ⚠️ 关键:用字符串构造,避免 double 二进制精度误差
.setScale(2, RoundingMode.DOWN);
System.out.println(bd.doubleValue()); // 输出:19.86
// 或直接获取字符串:bd.toString() → "19.86"❗ 重要提示:务必使用 new BigDecimal(Double.toString(x)) 而非 new BigDecimal(x),否则可能因 double 的二进制表示引入意外精度偏差(例如 0.1 可能变成 0.10000000000000000555...)。
? 总结建议
- 一般显示场景:优先选 DecimalFormat,轻量且直观;
- 精度敏感场景(如金额、科学计算):必须用 BigDecimal,并始终通过字符串构造;
- 避免陷阱:切勿依赖 Math.floor(x * 100) / 100 等浮点运算,因 double 本身存在精度缺陷,可能导致错误结果(如 19.867 * 100 实际为 1986.6999999999998);
- 所有方案均实现 向零截断(RoundingMode.DOWN),对负数同样适用(如 -19.867 → -19.86)。










