<วัตถุ Django> ไม่สามารถต่ออนุกรม JSON ได้


106

ฉันมีรหัสต่อไปนี้สำหรับการจัดลำดับชุดแบบสอบถาม

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 และคำสั่ง ความคิดใด ๆ ?



คำตอบ:


117

simplejsonและjsonไม่ทำงานกับวัตถุ django ได้ดี

อนุกรมในตัวของ Django สามารถทำให้เป็นอนุกรมเคียวรีที่เต็มไปด้วยวัตถุ django เท่านั้น

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

ในกรณีของคุณself.get_queryset()มีวัตถุ django และคำสั่งผสมอยู่ภายใน

ทางเลือกหนึ่งคือการกำจัดอินสแตนซ์โมเดลใน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

วิธีแก้ปัญหาดังกล่าวจะทำให้เกิดการค้นหาจำนวนมาก
Julio Marins

หากต้องการใช้สิ่งนั้นโดยตรงใน JS เพียงแค่ใช้safetage stackoverflow.com/a/57939897/4157431
Rami Alloush

66

วิธีที่ง่ายที่สุดคือการใช้JsonResponse

สำหรับ Queryset คุณควรส่งรายการของvaluesQueryset นั้นดังนี้:

from django.http import JsonResponse

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

2
ขอบคุณสำหรับ .values ​​() ในกรณีของฉันฉันแค่ต้องเพิ่ม. value () หลังตัวกรอง
Jze

20

ฉันพบว่าสามารถทำได้ค่อนข้างง่ายโดยใช้เมธอด ".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
ทางออกที่ง่ายและดีที่สุด
Timur

13

จากเวอร์ชัน 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)
Pablo Díaz

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

สิ่งนี้ใช้ได้ดีทีเดียว

โดยการใช้ไซต์ของเรา หมายความว่าคุณได้อ่านและทำความเข้าใจนโยบายคุกกี้และนโยบายความเป็นส่วนตัวของเราแล้ว
Licensed under cc by-sa 3.0 with attribution required.