如何在 Python 中精确截断浮点数至小数点后两位(不四舍五入)

本文介绍多种纯 python 方法(无需第三方库)将浮点数如 `68.82809999` 截断为 `68.82`,重点区分「截断」与「四舍五入」,涵盖字符串切片、数学运算和 decimal 精确控制等可靠方案。

在 Python 中,将浮点数「截断」(truncation)至小数点后两位(即直接丢弃后续小数位,而非四舍五入),是金融计算、价格显示等场景中的常见需求。例如输入 68.82809999,期望输出 68.82 —— 注意:这不是 round(68.82809999, 2) 的结果(后者为 68.83),而是严格向零截断。

以下是三种推荐的纯 Python 实现方式(均无需安装额外库):

✅ 方法一:乘法 + math.trunc() + 除法(推荐|数值安全)

import math

def truncate_to_2_decimals(x):
    return math.trunc(x * 100) / 100.0

print(truncate_to_2_decimals(68.82809999))  # 输出: 68.82
print(truncate_to_2_decimals(-3.14159))      # 输出: -3.14(负数也正确截断)

✅ 优点:逻辑清晰、支持正负数、避免浮点字符串精度陷阱;
⚠️ 注意:math.trunc() 向零取整,符合截断语义(区别于 int() 对负数的隐式截断行为)。

✅ 方法二:字符串切片(简洁直观,适合展示场景)

def truncate_str_method(x):
    s = str(x)
    if '.' in s:
        integer, decimal = s.split('.', 1)
        truncated_decimal = decimal[:2]  # 取前两位小数
        return float(f"{integer}.{truncated_decimal}")
    return float(x)

print(truncate_str_method(68.82809999))  # 输出: 68.82

✅ 优点:易理解、无导入依赖;
⚠️ 注意:对极小/极大数(如 1e-5)或科学计数法表示(如 6.8828e1)可能失效,不建议用于高精度或通用数值处理

✅ 方法三:decimal.Decimal(最高精度,推荐用于金融计算)

from decimal import Decimal, ROUND_DOWN

def truncate_decimal(x):
    d = Decimal(str(x))  # 先转为字符串再转Decimal,避免float构造误差
    return float(d.quantize(Decimal('0.01'), rounding=ROUND_DOWN))

print(truncate_decimal(68.82809999))  # 输出: 68.82
print(truncate_decimal(68.82000000))  # 输出: 68.82(保持末尾零语义)

✅ 优点:完全规避二进制浮点误差,支持可控舍入策略(ROUND_DOWN 即截断);
⚠️ 注意:需 str(x) 构造 Decimal,避免 Decimal(68.82809999) 因原始 float 已失真。

❌ 不推荐的做法说明

  • 使用 int(x * 100) / 100.0:对负数会错误向下取整(如 -3.14159 → int(-314.159) = -315 → -3.15);
  • 使用 math.floor(x * 100) / 100.0:对负数过度截断(floor(-314.159) = -315),不符合“向零截断”预期;
  • 直接 f"{x:.2f}":这是四舍五入格式化,非截断(68.828 → "68.83")。

总结

  • 日常使用 → 选 方法一(math.trunc);
  • 需要绝对精度(如金额)→ 选 方法三(Decimal + ROUND_DOWN);
  • 快速原型/仅处理简单正数 → 方法二可临时使用,但务必测试边界值。
    无论哪种方式,请始终明确「截断 ≠ 四舍五入」,并在关键业务中添加单元测试验证负数与边界情况(如 0.009, -0.009, 100.0)。