您好,登錄后才能下訂單哦!
本文環境
fix(2018.5.19):最近得知Django 的model基類需要聲明為abstract,故在原來的代碼加入abstract聲明,以免誤導
在Django中,關于如何將model類序列化為json,一般的話有兩a器
將model類轉為字典,再使用json庫的dumps方法轉為json
第一種方法就不多講了,直接去看官方文檔就好啦
一般來說,官方提供的方法應該都是比較好用和穩定的,然而,使用官方的序列化器卻問題不少:
格式丑陋,格式如下,一言難盡:
[ { "pk": "4b678b301dfd8a4e0dad910de3ae245b", "model": "sessions.session", "fields": { "expire_date": "2013-01-16T08:16:59.844Z", ... } } ]
是的,其中pk指的是默認主鍵,model指的是該object的model類型,然后fields才是obj的各種字段...真的是不知如何評價了
數了一通官方序列化器的缺點,當然了,上面的幾個點肯定是有解決方案的,但是啊,我確實不想折騰了嚶嚶嚶。
于是扔出我的解決方案:
關于生成object的字典的策略是這樣的:
源碼及使用方法
from django.db import models import json class BaseModel(models.Model): class Meta: abstract = True # 返回self._meta.fields中沒有的,但是又是需要的字段名的列表 # 形如['name','type'] def getMtMField(self): pass # 返回需要在json中忽略的字段名的列表 # 形如['password'] def getIgnoreList(self): pass def isAttrInstance(self, attr, clazz): return isinstance(getattr(self, attr), clazz) def getDict(self): fields = [] for field in self._meta.fields: fields.append(field.name) d = {} import datetime for attr in fields: if isinstance(getattr(self, attr), datetime.datetime): d[attr] = getattr(self, attr).strftime('%Y-%m-%d %H:%M:%S') elif isinstance(getattr(self, attr), datetime.date): d[attr] = getattr(self, attr).strftime('%Y-%m-%d') # 特殊處理datetime的數據 elif isinstance(getattr(self, attr), BaseModel): d[attr] = getattr(self, attr).getDict() # 遞歸生成BaseModel類的dict elif self.isAttrInstance(attr, int) or self.isAttrInstance(attr, float) \ or self.isAttrInstance(attr, str): d[attr] = getattr(self, attr) # else: # d[attr] = getattr(self, attr) mAttr = self.getMtMField() if mAttr is not None: for m in mAttr: if hasattr(self, m): attlist = getattr(self, m).all() l = [] for attr in attlist: if isinstance(attr, BaseModel): l.append(attr.getDict()) else: dic = attr.__dict__ if '_state' in dic: dic.pop('_state') l.append(dic) d[m] = l # 由于ManyToMany類不能存在于_meat.fields,因而子類需要在getMtMFiled中返回這些字段 if 'basemodel_ptr' in d: d.pop('basemodel_ptr') ignoreList = self.getIgnoreList() if ignoreList is not None: for m in ignoreList: if d.get(m) is not None: d.pop(m) # 移除不需要的字段 return d def toJSON(self): import json return json.dumps(self.getDict(), ensure_ascii=False).encode('utf-8').decode()
使用方法:
models的所有類都繼承BaseModel類,然后調用此類的toJSON()方法即可
注意,不知為何,self._meta.fields中沒有包含ManyToManyField字段,因而需要重寫getMtMField方法。例子如下:
class Book(BaseModel): name = models.CharField(max_length=50) authors = models.ManyToManyField(Author) publish = models.ForeignKey(Publisher, on_delete=models.SET_NULL, blank=True, null=True) page = models.IntegerField(default=0) # 頁數 introduction = models.CharField(max_length=500) bookType = models.ManyToManyField(BookType, null=True, blank=True) bookTag = models.ManyToManyField(BookTag, null=True, blank=True) evaluation = models.FloatField() coverUrl = models.CharField(max_length=100, null=True, blank=True) def getMtMField(self): return ['bookType', 'bookTag']
結果:
{ "id":4, "name":"Django從入門到放棄", "page":123, "introduction":"introduction", "evaluation":1, "bookType":[ { "id":1, "name":"類型" } ], "bookTag":[ { "id":2, "name":"tag" } ] }
后記
源碼有引用,即getDict方法中的第一個for循環,但懶得找原鏈接了,望見諒,特此聲明;
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持億速云。
免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。