Python对象转json【包括嵌套对象转json,django的model转json】
背景:
給app寫接口時經常會遇到將一個model轉為json返回。
問題:
網上也有類似方法,只是搜索結果多少有些問題,總是搜了好一會兒才找到合適的方法,另外,網上更多集中的只是簡單些的對象,對于復雜的對象,還是不容易找到好的方式。
方案(python3.6):
1對象轉json:
model類
class People():def __init__(self, name, age, pet):self.name = nameself.age = ageself.pet = pet class Pet():def __init__(self, pet_type, pet_name):self.pet_type = pet_typeself.pet_name = pet_name將Pet對象轉json:
import json def pet2json():pet = Pet('Cat', 'Lili')js = json.dumps(pet.__dict__)print(js)結果:{“pet_type”: “Cat”, “pet_name”: “Lili”}
小結:充分利用了Python對象的dict方法,Python下一切皆對象,每個對象都有多個屬性(attribute),Python對屬性有一套統一的管理方案。dict是用來存儲對象屬性的一個字典,其鍵為屬性名,值為屬性的值。dict可直接json化。
2嵌套對象轉json:
剛才的People類可看做是嵌套類,即有一個屬性是另一個類的實例,此時,若用上面的方法來json化Person對象,會有問題,如下【錯誤】:
def simple_person():pet = Pet('Cat', 'Lili')p = People('Xiaoming', 12,pet)json_data = json.dumps(p.__dict__)print(json_data)結果:報異常TypeError: Object of type ‘Pet’ is not JSON serializable
原因:json只能針對JSON serializable對象直接進行json化,而一般只有內置的類型,比如string,int,list和dict等才能直接序列化,代碼中p._ dict _是個dict類型,但是其pet屬性仍是自定義的類,是不能直接json化的。
解決方式【正確】:
結果:{“name”: “Xiaoming”, “age”: 12, “pet”: {“pet_type”: “Cat”, “pet_name”: “Lili”}}
小結:充分利用_ dict _方法。
3django的model轉json:
首先有個model類
class Person(models.Model):name = models.CharField(max_length=50, null=False)age = models.IntegerField(default=0)pid = models.CharField(max_length=20, unique=True)gender = models.IntegerField(default=0)針對該Person類,有兩種常見情況需要提供其json:
1:根據pid查詢person記錄;
2:根據某些條件,查詢一些person記錄。
此時數據庫里已經插入了一些數據
這里要提一下網上比較常見的一種方式,需要用到django.core.serializers,這個類的serialize(format, queryset, **options)方法,很明顯,這個方式只能作用與queryset格式,并且通過例子(不再列出),得到的結果類似這種 [{“model”: “polls.person”, “pk”: 2, “fields”: {“name”: “Cysion”, “age”: 29, “pid”: “3708261989”, “gender”: 0}}],出現了model,pk,field等屬性,不但用不到(對app來說),而且還增加了其它屬性的使用復雜度。這個在官網的說明文檔里也是如此處理,但是作者并不推薦。
方案:
我們還是使用_ dict _這個利器,首先,我們根據pid獲得一個Person對象,然后利用dict方法打印看看結果(錯誤)
req_pid=3708262007//request中得到try:rt = Person.objects.get(pid=req_pid)print(rt.__dict__)return HttpResponse(json.dumps(rt.__dict__),content_type='application/json')# return JsonResponse(rt.__dict__, safe=False)//另一種方式except:return JsonResponse(datalogic.get_comon_resp(1, '沒有查詢到對應數據'))結果是:print結果{‘_state’: < django.db.models.base.ModelState object at 0x0000000004C80860 >, ‘id’: 17, ‘name’: ‘zhaoliu’, ‘age’: 10, ‘pid’: ‘3708262007’, ‘gender’: 1},嘖嘖,又多了些屬性,特別是這個_state,是不能序列化的,所以上述并不能直接返回想要的結果。
作者:劉咸尚
解決方式:既然_state無用,且影響了結果,那我們直接臨時除去,不就返回了想要的結果嗎(正確)。
req_pid = request.POST.get('pid')try:rt = Person.objects.get(pid=req_pid)rt.__dict__.pop("_state")return JsonResponse(rt.__dict__, safe=False)except:return JsonResponse(datalogic.get_comon_resp(1, '沒有查詢到對應數據'))接口返回結果是:
{"id": 17,"name": "zhaoliu","age": 10,"pid": "3708262007","gender": 1 }正是客戶端需要的。
最后,是返回列表的,比如需要這種結果
{"code": 0,"msg": "成功","data": [{"id": 2,"name": "Cysion","age": 29,"pid": "3708261989","gender": 0},{"id": 11,"name": "Sophia","age": 22,"pid": "3708261998","gender": 1},{"id": 15,"name": "lisi","age": 13,"pid": "3708262005","gender": 0}] }實現思路同上面類似,首先數據庫查詢后得到QuerySet,其不能直接json化(通過serializer得到的不好看,也不好處理,大量的屬性處理還比較費勁),需要將其遍歷得到每個對象,然后將其屬性字典加入到list中,最后將其添加到通用dict中
pers = Person.objects.all() result = {"code":0,"msg":"成功"} L = [] for p in pers:p.__dict__.pop("_state")//需要除去,否則不能json化L.append(p.__dict__)//注意,實際是個json拼接的過程,不能直接添加對象 result ['data'] = L這個時候result 就是個可以直接json化的對象了,通過
return JsonResponse(result, safe=False)可以返回上面需要的結果。
主要提供了思路,詳細代碼就不提供了。
總結
以上是生活随笔為你收集整理的Python对象转json【包括嵌套对象转json,django的model转json】的全部內容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: 输入输出优化
- 下一篇: 太阳时的计算的Python程序