สำหรับผู้ที่ไม่ต้องการใช้ห้องสมุดบุคคลที่สาม ... ปัญหาของคำตอบของอีเลียสซามาเรียก็คือมันแปลงเป็นแบบลอยซึ่งสามารถพบปัญหาได้ ตัวอย่างเช่น:
>>> 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}'