Для тих, хто не хоче використовувати сторонню бібліотеку ... Проблема з відповіддю Еліаса Замарії полягає в тому, що вона перетворюється на плаваючий характер, що може зіткнутися з проблемами. Наприклад:
>>> 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 (як поплавок) , який потім буде закодований звичайним способом. Проблема в 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}'