Для тех, кто не хочет использовать стороннюю библиотеку ... Проблема с ответом Элиаса Замарии заключается в том, что она конвертируется в плавающую, что может привести к проблемам. Например:
>>> json.dumps({'x': Decimal('0.0000001')}, cls=DecimalEncoder)
'{"x": 1e-07}'
>>> json.dumps({'x': Decimal('100000000000.01734')}, cls=DecimalEncoder)
'{"x": 100000000000.01733}'
Этот JSONEncoder.encode()
метод позволяет вам возвращать буквальное содержимое json, в отличие от JSONEncoder.default()
которого вы возвращаете json-совместимый тип (например, float), который затем кодируется обычным способом. Проблема в encode()
том, что он (обычно) работает только на верхнем уровне. Но это все еще удобно, с небольшой дополнительной работой (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)
Что дает вам:
>>> json.dumps({'x': Decimal('0.0000001')}, cls=DecimalEncoder)
'{"x": 0.0000001}'
>>> json.dumps({'x': Decimal('100000000000.01734')}, cls=DecimalEncoder)
'{"x": 100000000000.01734}'