Python decimal.Decimal 以科学记数法生成结果

Python decimal.Decimal producing result in scientific notation

我将很长的数字分成更小的数字。两者都是 decimal.Decimal().

类型

结果以科学计数法出来。我该如何阻止呢?我需要完整打印号码。

>>> decimal.getcontext().prec
50
>>> val
Decimal('1000000000000000000000000')
>>> units
Decimal('1500000000')
>>> units / val
Decimal('1.5E-15')

内部保持精度 - 您只需在将十进制值导出到字符串时显式调用所需的小数位数。

因此,如果您要打印或在 HTML 模板中插入值,第一步是使用字符串格式化方法(或 f-strings),以确保包含数字:

In [29]: print(f"{units/val:.50f}")                                                                                                                    
0.00000000000000150000000000000000000000000000000000

不幸的是,string-format minilanguage 无法自行消除右侧多余的零。 (左边可以补“0”、“”、自定义字符,随便填,但小数点后的精度全部转为尾随0)。

因为找到最不重要的非零数字很复杂 - 否则我们可以使用从数字中提取的参数而不是“50”来提高格式表达式的精度,更简单的方法是在格式化后删除那些零发生,使用字符串 .rstrip 方法:


In [30]: print(f"{units/val:.50f}".rstrip("0"))                                                                                                        
0.0000000000000015

简而言之:这似乎是唯一的方法:在所有接口点,数字离开核心到输出,在那里它被表示为一个字符串,你用过高的精度格式化它定点符号,并用 f-string 去除尾部零:

return template.render(number=f"{number:.50f}".rstrip("0"), ...)

将小数转换为带有浮点类型指示符的格式化字符串 {:,f},无论它是非常大的整数还是非常大的小数。

>>> val
Decimal('1000000000000000000000000')

>>> units
Decimal('1500000000')

>>> "{:,f}".format(units / val)
'0.0000000000000015'

# very large decimal integer, formatted as float-type string, appears without any decimal places at all when it has none! Nice!

>>> "{:,f}".format(units * val)
 '1,500,000,000,000,000,000,000,000,000,000,000'

您不需要指定小数位。它将只显示表示数字所需的数量,当小数点短于固定格式宽度时,会忽略出现在最后一个小数位之后的无用零尾迹。如果数字没有小数部分,则不会得到任何小数位。

非常大的数字因此无需再猜测它们有多大就可以容纳。而且您也不必再猜测它们是否会有小数位。

任何指定的千位分隔符 {:,f}同样只有在数字是大整数而不是长小数时才有效。

附带条件

然而,

Decimal() 具有 重要位置 的想法,如果您认为需要,它会添加尾随零他们。

这个想法是它可以智能地处理您可能正在处理 £ 10.15 等货币数字的情况。要使用 documentation 中的示例:

>>> decimal.Decimal('1.30') +  decimal.Decimal('1.20')
Decimal('2.50')

如果您格式化 Decimal() 没有任何区别 - 如果 Decimal() 认为它很重要,您仍然会得到尾随零:

>>> "{:,f}".format( decimal.Decimal('1.30') +  decimal.Decimal('1.20'))
'2.50'

当你同时处理千位和分数时,同样的事情会发生(也许是出于某种充分的理由?):

>>> decimal.Decimal(2500) * decimal.Decimal('0.001')
Decimal('2.500')

使用 Decimal().normalize() 方法删除重要的尾随零

>>> (2500 * decimal.Decimal('0.001')).normalize()
Decimal('2.5')