Dla tych, którzy nie chcą korzystać z biblioteki innej firmy ... Problem z odpowiedzią Eliasa Zamarii polega na tym, że konwertuje się ona na float, co może powodować problemy. Na przykład:
>>> json.dumps({'x': Decimal('0.0000001')}, cls=DecimalEncoder)
'{"x": 1e-07}'
>>> json.dumps({'x': Decimal('100000000000.01734')}, cls=DecimalEncoder)
'{"x": 100000000000.01733}'
JSONEncoder.encode()
Metoda pozwala powrócić literalną treść json, w przeciwieństwie JSONEncoder.default()
, który wrócisz typ zgodny json (jak pływaka), który następnie zostaje zakodowany w normalny sposób. Problem encode()
polega na tym, że (normalnie) działa tylko na najwyższym poziomie. Ale nadal jest użyteczny, z odrobiną dodatkowej pracy (python 3.x):
import json
from collections.abc import Mapping, Iterable
from decimal import Decimal
class DecimalEncoder(json.JSONEncoder):
def encode(self, obj):
if isinstance(obj, Mapping):
return '{' + ', '.join(f'{self.encode(k)}: {self.encode(v)}' for (k, v) in obj.items()) + '}'
if isinstance(obj, Iterable) and (not isinstance(obj, str)):
return '[' + ', '.join(map(self.encode, obj)) + ']'
if isinstance(obj, Decimal):
return f'{obj.normalize():f}' # using normalize() gets rid of trailing 0s, using ':f' prevents scientific notation
return super().encode(obj)
Co daje ci:
>>> json.dumps({'x': Decimal('0.0000001')}, cls=DecimalEncoder)
'{"x": 0.0000001}'
>>> json.dumps({'x': Decimal('100000000000.01734')}, cls=DecimalEncoder)
'{"x": 100000000000.01734}'