<Об'єкт 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, так і з диктів. Якісь ідеї?


Відповіді:


114

simplejsonі jsonне працюйте добре з об’єктами джанго.

Вбудовані серіалізатори Django можуть лише серіалізувати набори запитів, наповнені об'єктами django:

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

У вашому випадку self.get_queryset()міститься суміш предметів джанго та диктів всередині.

Один із варіантів - позбутися екземплярів моделей у self.get_queryset()та замінити їх на дикти, використовуючи 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
Коли модель має поле дати та часу, вона не працює.
ax003d

це рішення спричинить багато запитів
Хуліо

щоб використовувати це безпосередньо в JS, просто використовуйте safetage. stackoverflow.com/a/57939897/4157431
Рамі Аллуш

62

Найпростіший спосіб - скористатися JsonResponse .

Для набору запитів ви повинні передати список valuesдля цього набору запитів, наприклад:

from django.http import JsonResponse

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

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" є лише символом, якщо його взяти як ітерабельний.

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


Це мені добре вдалось. Незважаючи на те, що повідомлення про помилку свідчить про те, що все це в одному великому списку, list()все одно, очевидно, потрібно.
trpt4him

1
Найпростіше і найкраще рішення
Тимур

11

З версії 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()))

Це працює досить добре

Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.