django rest_framework

安装:pip3 install djangorestframework

注意:如果django版本低于2.2,安装最新版,django也会会被更新到最新版

视图中使用:

from django.shortcuts import render

# Create your views here.

from rest_framework.views import APIView
from rest_framework.response import Response

from app01 import models

from app01.app01serializers import BookSer


class Home(APIView):

    def get(self, request, *args, **kwargs):
        res = [
            {"id": 1, "msg": "使用HTTPS传输数据"},
            {"id": 2, "msg": "api名称尽量使用名词复数,例如:Books、GirlFriends;不要使用:GetBook、Getgirl"},
            {"id": 3, "msg": "在API地址中要尽量标明这是一个API接口,例如:api.baidu。com、www。baidu。com/api"},
            {"id": 4, "msg": "多版本共存,更新api接口时,需要新建一个接口,而不是直接改原来的接口。例如:api.baidu.com/v1、api.baidu.com/v2"},
            {"id": 5, "msg": "查询条件在地址中携带,例如:api.baidu.com?title=xyj&limit=10"},
            {"id": 6, "msg": "对数据不同的操作方式,通过请求方法来体现,例如:获取GET、添加POST、删除DELETE、修改PUT"},
            {"id": 7, "msg": "获取多个数据返回列表,获取单个数据返回字典"},
            {"id": 8, "msg": "添加数据需要返回,添加数据的对象;修改数据需要返回被修改对象;删除数据返回空"},
            {"id": 9, "msg": "返回的数据中需要携带状态码,例如{code:200,msg:'成功'}"},
            {"id": 10,
             "msg": "在返回的数据中,携带下一次请求的连接,例如:{code:100,msg:成功,result:[{title:西游记,price:10,shop_url:baidu.com}]}"},
        ]

        return Response(res)


class Books(APIView):

    def get(self, request, *args, **kwargs):
        query_set = models.BookModel.objects.all()

        # 如果传入的是以一个queryset对象,必须传入many=True
        ser = BookSer(instance=query_set, many=True)

        return Response(ser.data)

    def post(self, request, *args, **kwargs):
        res = {
            'code': 200,
            'msg': '成功',
            'result': '',
            'errors': ''
        }
        # 使用APIView以后,rest_frameworkon 将所有数据都封装到了data中
        data = request.data

        ser = BookSer(data=data)
        if ser.is_valid():
            # 校验方式跟form一样,校验通过就调用save保存到数据库,需要注意,保存要自己在BookSer中重写create方法
            ser.save()
            res['result'] = ser.data
        else:
            res['code'] = 400
            res['msg'] = '失败'
            res['errors'] = ser.errors

        return Response(res)


class Book(APIView):

    def get(self, request, id, *args, **kwargs):

        book = models.BookModel.objects.filter(pk=id).first()

        ser = BookSer(instance=book)
        return Response(ser.data)

    def put(self, request, id, *args, **kwargs):
        res = {
            'code': 200,
            'msg': '成功',
            'result': '',
            'errors': ''
        }
        book = models.BookModel.objects.filter(pk=id).first()
        ser = BookSer(instance=book, data=request.data)
        if ser.is_valid():
            ser.save()
            res['result'] = ser.data

        else:
            res['code'] = 400
            res['msg'] = '失败'
            res['errors'] = ser.errors

        return Response(res)

    def delete(self, request, id, *args, **kwargs):
        res = {
            'code': 200,
            'msg': '成功',
            'result': '',
            'error': ''
        }
        try:
            book = models.BookModel.objects.filter(pk=id).delete()
        except Exception:
            res['error'] = str(Exception)

        return Response(res)

序列化使用:

from rest_framework import serializers
from rest_framework.exceptions import ValidationError

from app01 import models


class BookSer(serializers.Serializer):
    """
    read_only=False, 只读字段,用于显示。就算上传了也不会存入数据库
    write_only=False, 只写字段,用于保存,不会向前端从传递
    required=None, 是否可以为空,True可以 None不可以
    default=empty, 默认值,如果前端没有往后端传值,保存时会用默认值
    initial=empty,
    source=None,
    label=None, 标题
    help_text=None,提示信息
    style=None,
    error_messages=None, 错误提示
    validators=None, 字段校验,跟局部钩子类似,可以传递一个列表,列表中放多个校验函数,校验函数的写法跟局部钩子一样,函数名随意
    allow_null=False 是否可以传空,注意不是是否可以不传
    """
    title = serializers.CharField(label='书籍名称', max_length=32, min_length=2)
    price = serializers.FloatField(label='书籍价格', max_value=99999, min_value=1, read_only=True)
    publish = serializers.CharField(label='出版社名称', max_length=32, min_length=2)

    def validate_title(self, data):
        """
        局部钩子,用于校验字段是否合法,跟form的局部钩子类似
        :param data:
        :return:
        """
        if data.lower().startswith('sb'):
            raise ValidationError('书籍名称不能以SB开头')

        return data

    def validate(self, attrs):
        """
        全局钩子
        :param attrs:
        :return:
        """
        if self.instance: return attrs
        title = attrs.get('title')
        publish = attrs.get('publish')
        book = models.BookModel.objects.filter(title=title, publish=publish)
        if book:
            raise ValidationError('已存在相同书籍...')

        return attrs

    def create(self, validated_data):
        # 如果要保存,必须重写create方法
        obj = models.BookModel.objects.create(**validated_data)
        return obj

    def update(self, instance, validated_data):
        for k, v in validated_data.items():
            if hasattr(instance, k):
                setattr(instance, k, v)
        instance.save()
        return instance

 

posted on 2020-11-04 20:08  信奉上帝的小和尚  阅读(105)  评论(0编辑  收藏  举报

导航