Per coloro che non vogliono usare una libreria di terze parti ... Un problema con la risposta di Elias Zamaria è che si converte in float, il che può incorrere in problemi. Per esempio:
>>> json.dumps({'x': Decimal('0.0000001')}, cls=DecimalEncoder)
'{"x": 1e-07}'
>>> json.dumps({'x': Decimal('100000000000.01734')}, cls=DecimalEncoder)
'{"x": 100000000000.01733}'
Il JSONEncoder.encode()
metodo consente di restituire il contenuto json letterale, a differenza di quello JSONEncoder.default()
che restituisce un tipo compatibile json (come float) che viene quindi codificato in modo normale. Il problema encode()
è che (normalmente) funziona solo al livello più alto. Ma è ancora utilizzabile, con un po 'di lavoro extra (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)
Che ti dà:
>>> json.dumps({'x': Decimal('0.0000001')}, cls=DecimalEncoder)
'{"x": 0.0000001}'
>>> json.dumps({'x': Decimal('100000000000.01734')}, cls=DecimalEncoder)
'{"x": 100000000000.01734}'