16

我花了无数时间研究、阅读、测试,最终对 Python 的 Decimal 对象缺乏最基本的概念感到困惑和沮丧:将 Decimal 的输出格式化为字符串。

假设我们有一些具有以下值的字符串或 Decimal 对象:

   0.0008
  11.1111
 222.2222
3333.3333
1234.5678

目标是简单地将 Decimal 的精度设置为小数点后第二位。例如,11.1111将被格式化为11.11和。1234.56781234.57

我设想的代码类似于以下内容:

import decimal
decimals = [
  decimal.Decimal('0.0008'),
  decimal.Decimal('11.1111'),
  decimal.Decimal('222.2222'),
  decimal.Decimal('3333.3333'),
  decimal.Decimal('1234.5678'),
]
for dec in decimals:
  print dec.as_string(precision=2, rounding=ROUND_HALF_UP)

结果输出将是:

0.00
11.11
222.22
3333.33
1234.57

显然我们不能利用 Decimal 上下文的精度,因为这考虑了 TOTAL 位数,而不仅仅是小数精度。

我也对将 Decimal 转换为 float 以输出其值不感兴趣。Decimal 背后的全部原因是避免在浮点数上存储和运行计算。

还有哪些其他解决方案?我知道在堆栈溢出方面还有许多其他类似的问题,但我没有发现它们可以解决我正在询问的根本问题。

非常感谢!

4

2 回答 2

28

只需使用字符串格式format()函数

>>> for dec in decimals:
...    print format(dec, '7.2f')
... 
   0.00
  11.11
 222.22
3333.33
1234.57

decimal.Decimal支持与浮点数相同的格式规范,因此您可以根据需要使用指数、定点、通用、数字或百分比格式。

这是格式化小数的官方和pythonic方法;该类Decimal实现了.__format__()有效处理此类格式的方法。

于 2013-02-25T20:56:38.543 回答
2
def d(_in, decimal_places = 3):
    ''' Convert number to Decimal and do rounding, for doing calculations
        Examples:
          46.18271  to   46.183   rounded up
          46.18749  to   46.187   rounded down
          117.34999999999999 to 117.350
         _rescale is a private function, bad practice yet works for now.
    '''
    return Decimal(_in)._rescale(-decimal_places, 'ROUND_HALF_EVEN')

编辑:同样,_rescale() 并不打算被我们普通的两足动物使用,它在 Python 2.7 中工作,在 3.4 中不可用。

于 2014-03-13T06:29:08.003 回答