ModelSerializer序列化與反序列化(重點)

視圖類

class UserV3APIView(APIView):
    # 單查羣查
    def get(self, request, *args, **kwargs):
        pk = kwargs.get('pk')
        if pk:
            user_obj = models.User.objects.filter(is_delete=False, pk=pk).first()
            if not user_obj:
                return Response({
                    'status': 1,
                    'msg': 'pk error',
                }, status=400)

            user_ser = serializers.UserModelSerializer(user_obj, many=False)
            return Response({
                'status': 0,
                'msg': 'ok',
                'results': user_ser.data
            })
        else:
            user_query = models.User.objects.filter(is_delete=False).all()

            user_ser = serializers.UserModelSerializer(user_query, many=True)

            return Response({
                'status': 0,
                'msg': 'ok',
                'results': user_ser.data
            })

    # 單增
    def post(self, request, *args, **kwargs):
        user_ser = serializers.UserModelSerializer(data=request.data)
        if user_ser.is_valid():
            # 入庫
            user_obj = user_ser.save()
            return Response({
                'status': 0,
                'msg': 'ok',
                'results': serializers.UserModelSerializer(user_obj).data
            })
        else:
            return Response({
                'status': 1,
                'msg': user_ser.errors,
            })

序列化類

""" ModelSerializer類序列化與反序列化總結
1)序列化類繼承ModelSerializer,所以需要在配置類Meta中進行配置
2)model配置:綁定序列化相關的Model表
3)fields配置:採用 插拔式 設置所有參與序列化與反序列化字段
4)extra_kwargs配置:
    劃分系統字段爲三種:只讀(read_only)、只寫(write_only)、可讀可寫(不設置)
    字段是否必須:required
    選填字段:在extra_kwargs進行配置,但不設置required,且有默認值
5)自定義序列化字段:
    第一種(不提倡):在序列化類中用SerializerMethodField()來實現
    第二種(提倡):在模型類中用@property來實現,可插拔
6)自定義反序列化字段:
    同Serializer類,且規則只能在此聲明中設置,或是在鉤子中設置,在extra_kwargs中對其設置的無效
    自定義反序列化字段與系統字段,設置規則一樣,所以必須設置 write_only
7)局部鉤子,全局鉤子同Serializer類
8)不需要重寫create和update方法
"""
class UserModelSerializer(serializers.ModelSerializer):
    # 第一種自定義序列化字段:該字段必須在fields中設置
    # gender = serializers.SerializerMethodField()
    # def get_gender(self, obj):
    #     return obj.get_sex_display()


    # 自定義反序列化字段同Serializer類,且規則只能在此聲明中設置,或是在鉤子中設置,
    #       在extra_kwargs中對其設置的無效
    # 注:自定義反序列化字段與系統字段,設置規則一樣,所以必須設置 write_only
    re_password = serializers.CharField(min_length=3, max_length=16, write_only=True)

    class Meta:
        model = models.User
        # fields採用 插拔式 設置所有參與序列化與反序列化字段
        fields = ('username', 'gender', 'icon', 'password', 'sex', 're_password')
        extra_kwargs = {
            'username': {  # 系統字段不設置read_only和write_only,默認都參加
                'min_length': 3,
                'max_length': 10,
                'error_messages': {
                    'min_length': '太短',
                    'max_length': '太長'
                }
            },
            'gender': {
                'read_only': True,  # 自定義的序列化字段默認就是read_only,且不能修改,可以省略
            },
            'password': {
                'write_only': True,
            },
            'sex': {  # 像sex有默認值的字段,爲選填字段('required': True可以將其變爲必填字段)
                'write_only': True,
                # 'required': True
            }
        }


    # 局部全局鉤子同Serializer類,是與 Meta 同縮進的
    def validate_username(self, value):
        if 'g' in value.lower():
            raise serializers.ValidationError('名字中不能有g')
        return value

    def validate(self, attrs):
        password = attrs.get('password')
        re_password = attrs.pop('re_password')
        if password != re_password:
            raise serializers.ValidationError({'re_password': '兩次密碼不一致'})
        return attrs

    # create和update方法不需要再重寫,ModelSerializer類已提供,且支持所有關係下的連表操作

模型類

from django.db import models

class User(models.Model):
    SEX_CHOICES = (
        (0, '女'),
        (1, '男'),
    )

    username = models.CharField(max_length=64, verbose_name='用戶名', blank=True, unique=True)
    password = models.CharField(max_length=64, verbose_name='密碼')
    sex = models.IntegerField(choices=SEX_CHOICES, default=0, verbose_name='性別')
    img = models.ImageField(upload_to='img', default='img/default.png', verbose_name='頭像')
    # 開發中,數據不會直接刪除,通過字段控制
    is_delete = models.BooleanField(default=False, verbose_name='是否註銷')
    # 數據庫數據入庫,一般都會記錄該數據第一次入庫時間,有時候還會記錄最後一次更新時間
    created_time = models.DateTimeField(auto_now_add=True, verbose_name='創建時間')

    # 第二種自定義序列化字段(插拔式,提倡使用)
    @property
    def gender(self):
        return self.get_sex_display()

    @property
    def icon(self):
        from django.conf import settings
        return '%s%s%s' % (settings.BASE_URL, settings.MEDIA_URL, self.img)


    class Meta:  # 配置類,給所屬類提供配置信息
        db_table = 'old_boy_user'
        verbose_name_plural = '用戶表'

    def __str__(self):  # 不要在這裏進行連表操作,比如admin頁面可能會崩潰
        return self.username
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章