【问题标题】:How should I JSON-serialize Enum-children?我应该如何 JSON 序列化 Enum-children?
【发布时间】:2021-10-20 11:35:57
【问题描述】:

如果我写

import json

class Color():
    FUSCHIA = 0x00
    TURQUOISE = 0x01
    EMERALD = 0x02


def my_default_serializer(o):
    return o.__dict__


def get_colors():
    return json.dumps(
        {
            'dark_bg': True,
            'colors_batch': [Color.TURQUOISE,
                             Color.EMERALD]
        },
        default=my_default_serializer
    )

print(get_colors())

然后我得到了,正如预期的那样:

{"dark_bg": true, "colors_batch": [1, 2]}

现在假设我通过引入enum.Enum 对这段代码进行了现代化改造。除此之外,这很方便,因为Enum 将解析一个字符串(例如从文件中读取),因此我将上面的代码替换为:

import json
from enum import Enum

class Color(Enum):
    FUSCHIA = 0x00
    TURQUOISE = 0x01
    EMERALD = 0x02


def my_default_serializer(o):
    return o.__dict__


def get_colors():
    return json.dumps(
        {
            'dark_bg': True,
            'colors_batch': [Color['TURQUOISE'],
                             Color['EMERALD']]
        },
        default=my_default_serializer
    )

print(get_colors())

这引入了错误:

AttributeError: 'mappingproxy' object has no attribute '__dict__'

(如果我不指定默认序列化程序,我会得到TypeError: Object of type 'Color' is not JSON serializable )。

我应该如何对 Enum-children 进行 JSON 序列化?

【问题讨论】:

  • 你能分享big_obj的定义吗?为什么使用my_default_serializer
  • @balderman 感谢您的提问。我现在用一个自包含示例来说明问题的相同问题替换了这个问题。
  • 我设法序列化了Color。见下文

标签: python json python-3.x enums


【解决方案1】:

见下文 - 似乎有效。这个想法是覆盖JSONEncoderdefault方法,检查传入的参数是否是Color的实例并处理它。

import json
from enum import Enum


class Color(Enum):
    FUSCHIA = 0x00
    TURQUOISE = 0x01
    EMERALD = 0x02


class EnumEncoder(json.JSONEncoder):
    def default(self, obj):
        if isinstance(obj, Enum):
            return {"__enum__": str(obj)}
        return json.JSONEncoder.default(self, obj)


def get_colors():
    return json.dumps(
        {
            'dark_bg': True,
            'colors_batch': [Color['TURQUOISE'],
                             Color['EMERALD']]
        }, cls=EnumEncoder
    )


print(get_colors())

【讨论】:

  • 你的代码只适用于枚举Color——你能让它适用于任何枚举吗?
  • @EthanFurman - 添加了任何枚举支持。
  • 就是这样。只有一条评论:我们现在得到{"dark_bg": true, "colors_batch": [{"__enum__": "Color.TURQUOISE"}, {"__enum__": "Color.EMERALD"}]} 而不是{"dark_bg": true, "colors_batch": [1, 2]}。为了保持输出不变(这样我们就不必修改“另一边”,将return {"__enum__": str(obj)}替换为return obj.value
猜你喜欢
  • 2021-12-01
  • 1970-01-01
  • 2014-06-23
  • 1970-01-01
  • 2014-02-14
  • 2018-06-07
  • 2011-02-22
  • 1970-01-01
  • 1970-01-01
相关资源
最近更新 更多