<Objek Django> tidak dapat bersambung JSON


102

Saya memiliki kode berikut untuk serialisasi queryset;

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

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

Dan berikut adalah milik saya 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',.........]}]

Yang saya butuhkan untuk membuat serial. Tetapi dikatakan tidak dapat membuat serial <Product: hederello ()>. Karena list terdiri dari obyek dan dicts django. Ada ide?


Jawaban:


114

simplejsondan jsontidak bekerja dengan objek django dengan baik.

Django built-in serializers hanya bisa cerita bersambung querysets diisi dengan Django objek:

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

Dalam kasus Anda, self.get_queryset()berisi campuran obyek dan dicts di dalamnya.

Salah satu opsinya adalah menyingkirkan contoh model di self.get_queryset()dan menggantinya dengan dicts menggunakan 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")

Semoga membantu.


Sekarang mendapatkan kesalahan -> 'NoneType' object has no attribute 'concrete_model'... Dan menggunakan Django 1.4+
tuna

3
Jika model memiliki bidang datetime, itu tidak berfungsi.
ax003d

solusi itu akan memicu banyak pertanyaan
Julio Marins

untuk menggunakannya secara langsung di JS, cukup gunakan safetage. stackoverflow.com/a/57939897/4157431
Rami Alloush

62

Cara termudah adalah dengan menggunakan JsonResponse .

Untuk queryset, Anda harus meneruskan daftar valuesqueryset tersebut, seperti:

from django.http import JsonResponse

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

2
terima kasih untuk .values ​​(), Dalam kasus saya, saya hanya perlu menambahkan .values ​​() setelah filter
Jze

18

Saya menemukan bahwa ini dapat dilakukan dengan cukup sederhana menggunakan metode ".values", yang juga memberikan field bernama:

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

"list" harus digunakan untuk mendapatkan data sebagai iterable, karena jenis "value queryset" hanya berupa dict jika diambil sebagai iterable.

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


Ini bekerja dengan baik untuk saya. Meskipun pesan kesalahan menunjukkan semuanya dalam satu daftar besar, list()tampaknya masih diperlukan.
trpt4him

1
Solusi paling sederhana dan terbaik
Timur

11

Dari versi 1.9 Cara yang lebih mudah dan resmi untuk mendapatkan json

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


return JsonResponse(  model_to_dict(modelinstance) )

8

Js-programmer kami meminta saya untuk mengembalikan data format JSON yang tepat alih-alih string yang dikodekan json kepadanya.

Di bawah ini solusinya. (Ini akan mengembalikan objek yang dapat digunakan / dilihat langsung di browser)

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

Akan lebih baik hanyaHttpResponse(tmpObj)
Pablo Díaz

6

Pertama saya menambahkan metode to_dict ke model saya;

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

Lalu saya punya ini;

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)

dan terakhir gunakan kelas ini untuk membuat serialisasi queryset saya.

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

Ini bekerja dengan cukup baik

Dengan menggunakan situs kami, Anda mengakui telah membaca dan memahami Kebijakan Cookie dan Kebijakan Privasi kami.
Licensed under cc by-sa 3.0 with attribution required.