<Django object> 无法被转化为 JSON 格式。

168

我有以下代码用于序列化查询集:

def render_to_response(self, context, **response_kwargs):

    return HttpResponse(json.simplejson.dumps(list(self.get_queryset())),
                        mimetype="application/json")

以下是我的get_quersety()函数。

[{'product': <Product: hederello ()>, u'_id': u'9802', u'_source': {u'code': u'23981', u'facilities': [{u'facility': {u'name': {u'fr': u'G\xe9n\xe9ral', u'en': u'General'}, u'value': {u'fr': [u'bar', u'r\xe9ception ouverte 24h/24', u'chambres non-fumeurs', u'chambres familiales',.........]}]

我需要将其序列化。但它说无法序列化<Product: hederello ()>。因为该列表由Django对象和字典组成。有什么想法吗?


1
请勿重复提交:https://dev59.com/fmEh5IYBdhLWcg3w3muK#29088221 - Julio Marins
这个回答解决了你的问题吗?将Django模型对象转换为包含所有字段的字典 - Matej J
9个回答

168
simplejsonjson 不能很好地处理Django对象。 Django内置的序列化工具只能序列化填充有Django对象的查询集。
data = serializers.serialize('json', self.get_queryset())
return HttpResponse(data, content_type="application/json")

在您的情况下,self.get_queryset() 包含了 Django 对象和字典。

一个选择是通过使用 model_to_dict,将 self.get_queryset() 中的模型实例替换为字典。

from django.forms.models import model_to_dict

data = self.get_queryset()

for item in data:
   item['product'] = model_to_dict(item['product'])

return HttpResponse(json.simplejson.dumps(data), mimetype="application/json")

现在出现错误 --> 'NoneType' object has no attribute 'concrete_model' ... 并且使用的是 Django 1.4+。 - tuna
3
当模型具有日期时间字段时,它无法工作。 - ax003d
1
那个解决方案会触发很多查询。 - Julio Marins
要在JS中直接使用它,只需使用safe标签即可。https://dev59.com/Umw05IYBdhLWcg3wkik-#57939897 - Rami Alloush
还有回退的方法吗?另外,如果数据中有更多的项目,您的解决方案将覆盖产品。 - Macilias

122

最简单的方法是使用JsonResponse

对于查询集,您应该传递一个包含该查询集的values列表,如下所示:

from django.http import JsonResponse

queryset = YourModel.objects.filter(some__filter="some value").values()
return JsonResponse({"models_to_return": list(queryset)})

9
感谢使用.values()方法。在我的情况下,我只需在filter后面添加.values()即可。 - lwin

28

我发现可以使用".values"方法来实现这个操作,该方法还可以给出命名字段:

result_list = list(my_queryset.values('first_named_field', 'second_named_field'))
return HttpResponse(json.dumps(result_list))

"list"必须用于获取可迭代的数据,因为“值查询集”类型只有在作为可迭代对象时才会变成字典。

文档: https://docs.djangoproject.com/en/1.7/ref/models/querysets/#values


这对我很有效。尽管错误信息表明所有内容都在一个大列表中,但似乎仍然需要 list() - trpt4him

26

从1.9版本开始,获取JSON的方式更加简便和官方化。

from django.http import JsonResponse
from django.forms.models import model_to_dict


return JsonResponse(  model_to_dict(modelinstance) )

14

使用模型解决它的另一种绝佳方式是使用 values() 函数。

def returnResponse(date):
    response = ScheduledDate.objects.filter(date__startswith=date).values()
    return Response(response)


13
我们的JS程序员要求我向她返回确切的JSON格式数据,而不是一个JSON编码的字符串。
以下是解决方案。(这将返回一个对象,在浏览器中可以直接使用/查看)
import json
from xxx.models import alert
from django.core import serializers

def test(request):
    alert_list = alert.objects.all()

    tmpJson = serializers.serialize("json",alert_list)
    tmpObj = json.loads(tmpJson)

    return HttpResponse(json.dumps(tmpObj))

最好只用HttpResponse(tmpObj) - Pablo Díaz

9

首先,我给我的模型添加了一个to_dict方法;

def to_dict(self):
    return {"name": self.woo, "title": self.foo}

然后我有这个;
class DjangoJSONEncoder(JSONEncoder):

    def default(self, obj):
        if isinstance(obj, models.Model):
            return obj.to_dict()
        return JSONEncoder.default(self, obj)


dumps = curry(dumps, cls=DjangoJSONEncoder)

最后,使用这个类来序列化我的查询集。
def render_to_response(self, context, **response_kwargs):
    return HttpResponse(dumps(self.get_queryset()))

这很有效。

1

对于Django模型,请尝试:

users = User.objects.all()   
return JsonResponse ({'data' : list(users)}) 

0
你可以在Django模型中使用这个。这里我们通过safe=False来去掉包装器,否则会出错。但是.values()会以多维数组的形式返回key->value

views.py

from django.http import JsonResponse

users = User.objects.all().values()
return JsonResponse(list(users), safe=False)

1
请记住,Stack Overflow 不仅仅是为了解决当前的问题,还要帮助未来的读者找到类似问题的解决方案,这需要理解底层代码。对于我们社区中的初学者来说,这尤为重要,因为他们可能不熟悉语法。鉴于此,你能否编辑你的答案,包括解释你正在做什么以及为什么你认为这是最佳方法? - undefined

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接