本文环境
- Python 3.6.5
- Django 2.0.4
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的各种字段...真的是不知如何评价了
- 不能很好地支持list
- 对于一些外键(包括ManyToManyField等)不是很友好
- 甚至对于自身的DateField也没有很好的支持
数了一通官方序列化器的缺点,当然了,上面的几个点肯定是有解决方案的,但是啊,我确实不想折腾了嘤嘤嘤。
于是扔出我的解决方案:
- 新建一个类BaseModel,此类继承于官方的model类django.db.models.Model
- 在着个BaseModel中,声明一个方法,此方法用于生成关于这个object的字典
- 使用这个object的字典生成json
关于生成object的字典的策略是这样的:
- 通过反射获取这个object的所有字段名
- 根据字段名获得某个字段field
- 如果filed的类型的是int、float、str的话,直接将以 "字段名":字段值 的形式放入字典中
- 若field的类型是datetime或者date的话,使用date的方式处理,然后放入字典
- 若field的类型是BaseModel的话,那么就调用该field的getDict方法递归获得该field对应的字典,然后放入字典中
- 若field的类型是ManyToMany类型,在具体草种中我们使用这个field的all方法来这个field的所有object,然后也是通过getDict方法将其放入到字典中
源码及使用方法
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循环,但懒得找原链接了,望见谅,特此声明;
- 本人python新手,代码多有不规范之处,望见谅;
- 代码不精,但是也希望能帮到你_;
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。
更新日志
- 小骆驼-《草原狼2(蓝光CD)》[原抓WAV+CUE]
- 群星《欢迎来到我身边 电影原声专辑》[320K/MP3][105.02MB]
- 群星《欢迎来到我身边 电影原声专辑》[FLAC/分轨][480.9MB]
- 雷婷《梦里蓝天HQⅡ》 2023头版限量编号低速原抓[WAV+CUE][463M]
- 群星《2024好听新歌42》AI调整音效【WAV分轨】
- 王思雨-《思念陪着鸿雁飞》WAV
- 王思雨《喜马拉雅HQ》头版限量编号[WAV+CUE]
- 李健《无时无刻》[WAV+CUE][590M]
- 陈奕迅《酝酿》[WAV分轨][502M]
- 卓依婷《化蝶》2CD[WAV+CUE][1.1G]
- 群星《吉他王(黑胶CD)》[WAV+CUE]
- 齐秦《穿乐(穿越)》[WAV+CUE]
- 发烧珍品《数位CD音响测试-动向效果(九)》【WAV+CUE】
- 邝美云《邝美云精装歌集》[DSF][1.6G]
- 吕方《爱一回伤一回》[WAV+CUE][454M]