<Django nesnesi> JSON serileştirilebilir değil


106

Sorgu kümesini serileştirmek için aşağıdaki koda sahibim;

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

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

Ve takip etmek benim 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',.........]}]

Bunu serileştirmem gerekiyor. Ancak <Product: hederello ()>. Çünkü liste hem django nesnelerinden hem de diktelerden oluşur. Herhangi bir fikir ?



Yanıtlar:


117

simplejsonve jsondjango nesneleriyle iyi çalışmaz.

Django'nun yerleşik serileştiricileri yalnızca django nesneleriyle dolu sorgu kümelerini serileştirebilir:

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

Sizin durumunuzda, self.get_queryset()içinde django nesnelerinin ve diktlerin bir karışımını içerir.

Seçeneklerden biri, içindeki model örneklerinden kurtulup self.get_queryset()bunları kullanarak bunları diktlerle değiştirmektir 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")

Umarım yardımcı olur.


Şimdi hata alıyorum -> 'NoneType' object has no attribute 'concrete_model'... Ve Django 1.4+ kullanılıyor
tuna

3
Model bir tarih saat alanına sahip olduğunda çalışmaz.
ax003d

bu çözüm birçok
Julio Marins

doğrudan JS'de kullanmak için safetage kullanın . stackoverflow.com/a/57939897/4157431
Rami Alloush

66

En kolay yol bir JsonResponse kullanmaktır .

Bir sorgu kümesi için, valueso sorgu kümesinin bir listesini şu şekilde iletmelisiniz:

from django.http import JsonResponse

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

2
Değerler, () için teşekkürler, Benim durumumda, ben sadece filtreden sonra Değerler, () eklemek gerekir
Jze

20

Bunun, adlandırılmış alanlar da veren ".values" yöntemi kullanılarak oldukça basit bir şekilde yapılabileceğini buldum:

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

Verileri yinelenebilir olarak almak için "liste" kullanılmalıdır, çünkü "değer sorgu kümesi" türü yalnızca yinelenebilir olarak alındığında bir diktedir.

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


Bu benim için iyi çalıştı. Hata mesajı hepsinin tek bir büyük listede olduğunu öne sürse de, list()görünüşe göre hala gerekli.
trpt4him

1
En basit ve en iyi çözüm
Timur

13

1.9 sürümünden itibaren json almanın daha kolay ve resmi yolu

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


return JsonResponse(  model_to_dict(modelinstance) )

8

Js programcımız, ona json kodlu bir dize yerine tam JSON formatındaki verileri döndürmemi istedi.

Çözüm aşağıdadır. (Bu, tarayıcıda doğrudan kullanılabilen / görüntülenebilen bir nesne döndürür)

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))

Daha iyi olacakHttpResponse(tmpObj)
Pablo Díaz

6

Önce modelime bir to_dict yöntemi ekledim;

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

O zaman bu var;

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)

ve sonunda bu sınıfı sorgu kümemi serileştirmek için kullanın.

def render_to_response(self, context, **response_kwargs):
    return HttpResponse(dumps(self.get_queryset()))

Bu oldukça iyi çalışıyor

Sitemizi kullandığınızda şunları okuyup anladığınızı kabul etmiş olursunuz: Çerez Politikası ve Gizlilik Politikası.
Licensed under cc by-sa 3.0 with attribution required.