专题四:类视图

本章节介绍基于DRF类视图,以Host的增删改差的功能实现,分别介绍APIView,Mixins,Generic三个类视图介绍。APIView,Mixins,Generic之间的关系:最基础的是APIView。Mixins,Generic分别对Mixins,Generic封装

精华

image.png
image.png

通用代码

本文定义host的model和序列化,示例代码如下:

# model.py
class Host(models.Model):
    """
    域名
    """
    name = models.CharField(max_length=50, verbose_name="名称")
    host = models.CharField(max_length=1024, verbose_name="host地址")
    description = models.CharField(max_length=1024, blank=True, null=True, verbose_name="描述")
    project = models.ForeignKey(Project, on_delete=models.CASCADE, verbose_name="项目归属", related_name="host_list")
    create_time = models.DateTimeField(auto_now_add=True, null=True, verbose_name="创建时间")

# serialzers.py
class HostSerializer(serializers.ModelSerializer):
    class Meta:
        model = models.Host
        # __all__会把数据库对应的字段都返回
        # 包括id,host的model对应外键的project以project存储和返回
        fields = "__all__"

APIView

APIView是DRF中类视图最基础的父类。用法与django中的view一致,自己分别实现get,post,put,delete等方法。

1.get请求:根据传参pk是否为空,实现目标host查询和所有host获取。需要定义两个路由来实现,具体参考示例的url.py。
2.post请求:新建host记录,models.objects.create(必须字段),然后序列化返回给前端
3.put请求:更新host记录,序列化的优势就出来了:
赋值serializer = serializers.HostSerializer(host, data=request.data), host为修改的模型对象,request.data为更新的参数
校验:serializer.is_valid():
保存:serializer.save()
4.delete请求:删除对应目标

目标host查询

url配置为host/apiview/<int:pk>,其中pk为默认的参数变量,与类视图get(self, request, pk)要保持一致;

所有host获取

url配置为host/apiview

类视图view.py
class HostAPIView(views.APIView):
    permission_classes = [IsAuthenticated]
    authentication_classes = [JWTAuthentication]

    def get(self, request, pk=None):
        """
        pk!=None,查询单个;pk=None,获取所有
        """
        if pk:
            host = models.Host.objects.get(pk=pk)
            serializer = serializers.HostSerializer(host)
            return Response(serializer.data)
        else:
            host = models.Host.objects.all()
            serializer = serializers.HostSerializer(host, many=True)
            return Response(serializer.data)

    def post(self, request):
        """
        新建
        """
        host = models.Host.objects.create(
            name=request.data.get('name'),
            host=request.data.get('host'),
            description=request.data.get('description'),
            project_id=request.data.get('project'),
        )
        serializer = serializers.HostSerializer(host)
        return Response(serializer.data)

    def put(self, request, pk):
        """
        更新
        """
        host = models.Host.objects.get(pk=pk)
        serializer = serializers.HostSerializer(host, data=request.data)
        if serializer.is_valid():
            serializer.save()
            return Response(serializer.data)
        return Response(serializer.errors, status=status.HTTP_400_BAD_REQUEST)

    def delete(self, request, pk):
        """
        删除
        """
        host = models.Host.objects.get(pk=pk)
        host.delete()
        return Response(status=status.HTTP_204_NO_CONTENT)
路由代码
#urls.py
urlpatterns = [
    path('host/apiview', views_clas.HostAPIView.as_view(), name='host_api_view'),
    path('host/apiview/<int:pk>', views_clas.HostAPIView.as_view(), name='host_api_view'),
              ] + router.urls
APIView类视图属性:

authentication_classes :认证方式,默认 api_settings.DEFAULT_AUTHENTICATION_CLASSES
throttle_classes:限流策略,默认api_settings.DEFAULT_THROTTLE_CLASSES
permission_classes:权限控制,默认api_settings.DEFAULT_PERMISSION_CLASSES

Mixins

mixins是组件的意思,在DRF中,针对增,删,改,查都有对应的mixins。Mixins是对APIView的封装,即对应的请求类型方法还需要申明,但对应请求类型方法的具体实现根据mixins封装方法实现。
generics.GenericAPIView:基础类,其属性queryset对应数据对象,serializer_class对应的序列化
mixins.ListModelMixin:get请求,获取列表,与RetrieveModelMixin不能共存(暂用get请求类型),self.list(request)
mixins.RetrieveModelMixin:get请求,检索查询,self.retrieve(request)
mixins.CreateModelMixin:post请求,新建对象,self.create(request)
mixins.UpdateModelMixin:put请求,更新对象,self.update(request)
mixins.DestroyModelMixin:delete请求,删除对象,self.destroy(request)

示例代码如下:

class HostMixinsViews(
    generics.GenericAPIView,
    mixins.CreateModelMixin,
    mixins.RetrieveModelMixin,
    mixins.ListModelMixin,
    mixins.UpdateModelMixin,
    mixins.DestroyModelMixin
):
    queryset = models.Host.objects.all()
    serializer_class = serializers.HostSerializer

    def get(self, request, pk=None):
        if pk:
            return self.retrieve(request)
        else:
            return self.list(request)

    def post(self, request):
        return self.create(request)

    def put(self, request, pk=None):
        return self.update(request)

    def delete(self, request, pk=None):
        return self.destroy(request)

通过继承generics.GenericAPIView, 可以设置queryset以及serializer_class,类视图函数就知道要处理哪个模型和序列化。

路由配置Mixins与APIView一致:

rlpatterns = [
    # apiview
    path('host/apiview', views_clas.HostAPIView.as_view(), name='host_api_view'),
    path('host/apiview/<int:pk>', views_clas.HostAPIView.as_view(), name='host_api_view'),
    # mixins
    path('host/mixins', views_clas.HostMixinsViews.as_view(), name='host_mixins_view'),
    path('host/mixins/<int:pk>', views_clas.HostMixinsViews.as_view(), name='host_mixins_view'),
              ] + router.urls

Generic

Generic是在Mixins上的进一步封装,通过显示的基础generic的子类快速实现对象的增删改差。有以下generic类视图:

generics.ListAPIView:实现获取列表的。实现get方法。
generics.CreateAPIView:实现创建数据的。实现post方法。
generics.UpdateAPIView:实现更新数据的。实现put方法。
generics.DestroyAPIView:实现删除数据的。实现delete方法。
generics.RetrieveAPIView:实现检索数据的。
generics.ListCreateAPIView:实现列表和创建数据的。
generics.RetrieveUpdateAPIView:实现检索和更新数据的。
generics.RetrieveDestroyAPIView:实现检索和删除数据的。
generics.RetrieveUpdateDestroyAPIView:实现检索和更新和删除数据的。

Host示例代码如下:
class HostGenericViews(
    generics.CreateAPIView,
    generics.UpdateAPIView,
    generics.DestroyAPIView,
    generics.RetrieveAPIView
):
    queryset = models.Host.objects.all()
    serializer = serializers.HostSerializer
路由代码如下:
urlpatterns = [
                  # apiview
                  path('host/apiview', views_clas.HostAPIView.as_view(), name='host_api_view'),
                  path('host/apiview/<int:pk>', views_clas.HostAPIView.as_view(), name='host_api_view'),
                  # mixins
                  path('host/mixins', views_clas.HostMixinsViews.as_view(), name='host_mixins_view'),
                  path('host/mixins/<int:pk>', views_clas.HostMixinsViews.as_view(), name='host_mixins_view'),
                  # generics
                  path('host/generics', views_clas.HostMixinsViews.as_view(), name='host_generics_view'),
                  path('host/generics/<int:pk>', views_clas.HostMixinsViews.as_view(), name='host_generics_view'),
              ] + router.urls

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,142评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,298评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,068评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,081评论 1 291
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,099评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,071评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,990评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,832评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,274评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,488评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,649评论 1 347
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,378评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,979评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,625评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,643评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,545评论 2 352

推荐阅读更多精彩内容