34

我想使用 simplejson 序列化 Django 模型。Django 的序列化程序不支持字典……而 simplejson 不支持 Django 查询集。这是一个相当大的难题。

在模型中,赞助商有一个外键到赞助商级别,我试图将属于某个赞助商级别的所有赞助商分组在一起。这是生成列表的代码:

from django.shortcuts import get_list_or_404
from special_event.models import Sponsor, SponsorLevel

sponsor_dict = {}

roadie_sponsors = get_list_or_404(Sponsor, level__category = SponsorLevel.ROADIE_CHOICE)

for item in roadie_sponsors:
    try:
        sponsor_dict[item.level.name].append(item)
    except KeyError:
        sponsor_dict[item.level.name] = [item]

这是sponsor_dict“制造”后的样子

{
    'Fan': [<Sponsor: Fan Sponsor>], 
    'VIP': [<Sponsor: VIP Sponsor>],
    'Groupie': [<Sponsor: Groupie Sponsor>],
    'Silver': [<Sponsor: Silver Sponsor>],
    'Bronze': [<Sponsor: Another Bronze Sponsor>, <Sponsor: Bronze Sponsor>]
}

我只在每个级别添加了一个赞助商,除了铜牌,只是为了展示它是如何运作的。我想做的就是将它“全部”转换为 JSON,以便 jQuery 可以轻松地解释它。Django 的其他序列化程序(如 XML 或 YAML)可以做到这一点吗?我可以“扩展”Django JSON Serializer 来处理字典或“扩展”simplejson 来处理 Django QuerySet 对象吗?

4

3 回答 3

39

我会扩展simplejson。基本上,您想在 JSON 编码器遇到 QuerySet 时插入 django 的序列化。你可以使用类似的东西:

from json import dumps, loads, JSONEncoder

from django.core.serializers import serialize
from django.db.models.query import QuerySet
from django.utils.functional import curry

class DjangoJSONEncoder(JSONEncoder):
    def default(self, obj):
        if isinstance(obj, QuerySet):
            # `default` must return a python serializable
            # structure, the easiest way is to load the JSON
            # string produced by `serialize` and return it
            return loads(serialize('json', obj))
        return JSONEncoder.default(self,obj)

# partial function, we can now use dumps(my_dict) instead
# of dumps(my_dict, cls=DjangoJSONEncoder)
dumps = curry(dumps, cls=DjangoJSONEncoder)

有关default方法的更多信息,请查看simplejson 文档。把它放在一个 python 模块中,然后导入dumps,你就可以开始了。但请注意,此函数只会帮助您序列化QuerySet实例,而不是Model直接帮助您序列化实例。

于 2010-02-12T10:51:32.753 回答
13

在 django 中序列化大多数结构的一种非常灵活的方法是使用此处找到的序列化程序类

于 2011-07-03T13:57:26.027 回答
10

根据克莱门特的回答,我这样做也是为了将模型转换为 JSON。

def toJSON(obj):
   if isinstance(obj, QuerySet):
       return simplejson.dumps(obj, cls=DjangoJSONEncoder)
   if isinstance(obj, models.Model):
       #do the same as above by making it a queryset first
       set_obj = [obj]
       set_str = simplejson.dumps(simplejson.loads(serialize('json', set_obj)))
       #eliminate brackets in the beginning and the end 
       str_obj = set_str[1:len(set_str)-2]
   return str_obj
于 2010-08-14T20:36:58.317 回答