<Объект Django> не поддерживает сериализацию JSON

102

У меня есть следующий код для сериализации набора запросов;

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

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

И вот мой get_querset()

[{'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, так и из dicts. Любые идеи ?

тунец
источник
Дублировано: stackoverflow.com/a/29088221/2172260
Хулио Маринс,

Ответы:

116

simplejsonи jsonплохо работают с объектами django.

Встроенные сериализаторы Django могут сериализовать только наборы запросов, заполненные объектами django:

data = serializers.serialize('json', self.get_queryset())
return HttpResponse(data, content_type="application/json")

В твоем случае, self.get_queryset() внутри содержится смесь объектов django и dicts.

Один из вариантов - избавиться от экземпляров модели в self.get_queryset()и заменить их на dicts, используя model_to_dict:

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+
тунец
3
Когда в модели есть поле datetime, оно не работает.
ax003d
это решение вызовет множество запросов
Хулио Маринс
чтобы использовать это непосредственно в JS, просто используйте safeтег. stackoverflow.com/a/57939897/4157431
Рами
63

Самый простой способ - использовать JsonResponse .

Для набора запросов вы должны передать список valuesдля этого набора запросов, например:

from django.http import JsonResponse

queryset = YourModel.objects.filter(some__filter="some value").values()
return JsonResponse({"models_to_return": list(queryset)})
YPCrumble
источник
2
спасибо за .values ​​(), в моем случае мне просто нужно добавить .values ​​() после фильтра
Jze
18

Я обнаружил, что это можно сделать довольно просто, используя метод ".values", который также дает именованные поля:

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

«list» должен использоваться для получения данных как итерируемых, поскольку тип «value queryset» является только dict, если выбран как итеративный.

Документация: https://docs.djangoproject.com/en/1.7/ref/models/querysets/#values

Дэнни Стейпл
источник
У меня это сработало. Несмотря на то, что сообщение об ошибке предполагает, что все в одном большом списке, по- list()видимому, он все еще необходим.
trpt4him
1
Тимур
12

Начиная с версии 1.9 Более простой и официальный способ получения json

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


return JsonResponse(  model_to_dict(modelinstance) )
Яш
источник
8

Наш 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)
Пабло Диас
6

Сначала я добавил к своей модели метод 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()))

Это работает довольно хорошо

тунец
источник