django restframework serializer 增加自定義字段操作
在使用django restframework serializer 序列化在django中定義的model時,有時候我們需要額外在serializer中增加一些model中沒有的字段。有兩種方法實現(xiàn)這個目的。
假設(shè)現(xiàn)在有一個Animal模型,其中有name, type, country字段,country為外鍵。我們在序列化Animal時,需要額外增加外鍵country的area信息。
方法一修改數(shù)據(jù)庫,利用model 這里就不多解釋,主要來說第二種,不修改django的model,直接使用SerializerMethodField(method_name=None)字段。
class AnimalSerializer(serializers.ModelSerializer): country_area = serializers.SerializerMethodField() class Meta: model = Animal fields = ('id', 'name', 'type','country','country_area') def get_country_area(self, obj): return obj.country.area
SerializerMethodFiel是一個read-only字段
當(dāng)不指定其method_name時,默認為get_field_name
如果使用ModelSerializer并指定字段時,要包含此時定義的字段
補充知識:django restframework Serializer field
SerializerMethodField
這是一個只讀字段。它通過調(diào)用它所連接的序列化類的方法來獲得它的值。它可用于將任何類型的數(shù)據(jù)添加到對象的序列化表示中。
簽名: SerializerMethodField(method_name=None)
method_name - 要調(diào)用序列化對象的方法的名稱。如果不包含,則默認為 get_<field_name>.
由 method_name 參數(shù)引用的序列化方法應(yīng)該接受一個參數(shù)(除了 self),這是要序列化的對象。它應(yīng)該返回你想要包含在對象的序列化表示中的任何內(nèi)容。例如:
場景介紹:一個用戶往往對應(yīng)多個角色,而角色字段并不在UserProfile表中,這就需要我們新增角色字段到user序列化中
from .models import UserProfile from .models import UserRole class UserProfileSerializer(serializers.ModelSerializer): """ show list serializer """ role = serializers.SerializerMethodField() class Meta: model = UserProfile # fields = "__all__" fields = ["id", "username","role", "account", "really_name", "department", "tel_phone", "create_time", "email", "last_time", "creator"] def get_role(self,obj): user_id = obj.id roles = [i.role.name for i in UserRole.objects.filter(user_id=user_id)] roles = ",".join(roles) return roles
相反的場景:我們提交的表單數(shù)據(jù)存在于多表中(因為表中含有多對多字段),如何驗證所有字段,并保存完整的數(shù)據(jù)到各表中。
剛開始的思路在ModelSerializer中新增未定義字段,然后發(fā)現(xiàn)這并不可行。我也犯了SerializerMethodField的錯誤,但隨后去讀了serializer的源碼,以及了解serializer的順序,就知道SerializerMethodField僅用于list方法。
問題的突破點到底在哪里呢,在drf serializer 官方文檔中就有一個知識點,serializer.save(**kwargs),kwargs數(shù)據(jù)被綁定在serializer.validated_data對象上,當(dāng)create或update的時候就會被添加進數(shù)據(jù)庫,相當(dāng)于validated_data.update(kwargs)
實際源碼不是這樣子,但也就是這么個意思。
重寫Serializer create,update方法
from rest_framework.utils import model_meta def create(self, validated_data): field= validated_data.pop('field_name') validated_data = validated_data instance = Model.objects.create(**validated_data) # ...外鍵表的操作 return instance def update(self, instance, validated_data): field= validated_data.pop('field_name') info = model_meta.get_field_info(instance) for attr, value in validated_data.items(): if attr in info.relations and info.relations[attr].to_many: field = getattr(instance, attr) field.set(value) else: setattr(instance, attr, value) instance.save() # ... return instance
以上這篇django restframework serializer 增加自定義字段操作就是小編分享給大家的全部內(nèi)容了,希望能給大家一個參考,也希望大家多多支持腳本之家。
相關(guān)文章
Python爬蟲之Selenium實現(xiàn)關(guān)閉瀏覽器
這篇文章主要介紹了Python爬蟲之Selenium實現(xiàn)關(guān)閉瀏覽器,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-12-12Python內(nèi)置模塊hashlib、hmac與uuid用法分析
這篇文章主要介紹了Python內(nèi)置模塊hashlib、hmac與uuid用法,結(jié)合實例形式較為詳細的分析了hashlib、hmac與uuid模塊的概念、功能及簡單使用方法,需要的朋友可以參考下2018-02-02舉例區(qū)分Python中的淺復(fù)制與深復(fù)制
這篇文章主要介紹了舉例區(qū)分Python中的淺復(fù)制與深復(fù)制,是Python入門學(xué)習(xí)中的重要知識,需要的朋友可以參考下2015-07-07Pytorch-mlu?實現(xiàn)添加逐層算子方法詳解
本文主要分享了在寒武紀(jì)設(shè)備上?pytorch-mlu?中添加逐層算子的方法教程,代碼具有一定學(xué)習(xí)價值,有需要的朋友可以借鑒參考下,希望能夠有所幫助2021-11-11