万能的Python竟然算不明白“钱”?快来看一下

来自: Python那些事(微信号:PythonSomething),作者:蓝色波浪

金融行业的同学,经常会遇到“钱”的转换问题,在进行金额的计算和转化时,稍微有不慎,则会损失惨重。


1、带来经济损失,可能比较微小,

2、严重影响了后续的日终对账业务,

3、客户的投诉也会比较严重,严重影响公司的形象。


因而,在使用Python的时候,需要学会算明白“钱”。


先看一个例子,由“元”转换为“分”, Python代码如下:

>>> a = 66.9
>>> a * 100

6690.000000000001

咦,是不是令你大吃一惊?这么简单的运算Python竟然会有误?


确实转换为分之后竟然少了0.000000000001。这种微小的差别对于日终对账是致命的,可能运维人员需要排查一整晚,为了这小小的0.000000000001。有人可能说,这个问题简单,直接取整就可以了。代码如下:

>>> round(a * 100)

6690

事情真没有这么简单,金融计算对精度的要求不是一般高,四舍五入的操作同样会导致账目不平。


那么,如何Python如何计算“钱”呢?本文介绍两种方式。


使用定点数


其中一个方式是采用定点数。字面意思看,小数点位置是固定的,即约定机器中所有数据的小数点位置是不变的。这能够确保精度。

import decimal

>>> a = 66.9
>>> print(int(decimal.Decimal(str(a)) * decimal.Decimal(100)))
6690

使用库money


首先给出文档地址:

https://pypi.org/project/money/


money库,顾名思义,就是解决“钱”相关的问题。安装很简单,

pip install money

使用起来也比较简单,Money对象入参有两个,一个是数量,一个是货币单位。创建完Money对象,就可以做各种操作了:

from money import Money

m = Money(amount='66.9', currency='RMB')
print(m * 100)

遗憾的是,没提供内置函数来实现元和角、分的转换。


那Money对象的实现原理是什么呢?可以看一下Money对象的定义:

class Money(object):
   """Money class with a decimal amount and a currency"""

   def __init__(self, amount="0", currency=None):
       try:
           self._amount = decimal.Decimal(amount)
       except decimal.InvalidOperation:
           raise ValueError("amount value could not be converted to "
                            "Decimal(): '{}'".format(amount)) from None
       if currency in [None, False, '']:
           raise ValueError("invalid currency value: '{}'".format(currency))
       if not REGEX_CURRENCY_CODE.match(currency):
           raise ValueError("currency not in ISO 4217 format: "
                            "'{}'".format(currency))
       self._currency = currency

   @property
   def amount(self):
       return self._amount

   @property
   def currency(self):
       return self._currency

从代码中可以看出,Money对象有2个属性,一个是_amount,一个是_currency。其中,_amount在初始化时,也是将传入的值转化为定点数,其本质也是依赖于定点数。


所以,后续使用Python计算时,一定要用定点数,浮点数是算不明白“”的哦。

推荐↓↓↓
Python编程
上一篇:这些Python函数简直是屌爆了 下一篇:用Python打造属于自己的“今日头条” | 一个非常适合练手的全栈项目