drf中的各种view,viewset

Django REST framework里有各种各样的view,让我有点蒙,得好好捋一捋这关系。

视图的作用

Django用“视图”这个概念封装处理用户请求并返回响应的逻辑。

视图是一个可调用对象,它不仅可以是基于函数,也可以是基于类的。

相比较与函数,基于类的视图有一些区别和优势:

  • 组织与特定HTTP方法相关的代码(GET,POST等) 可以通过单独的方法而不是条件分支来解决。
  • 面向对象的技术例如Mixin(多继承、混用)可以将代码分解成可重用的组件。

视图函数:

def my_view(request):
if request.method == 'GET':
# <view logic>
return HttpResponse('result')

视图类:

class MyView(View):
def get(self, request):
# <view logic>
return HttpResponse('result')

View

Django中的View

是所有基于类的view的父类,它负责将视图连接到URL、HTTP 方法调度(GET,POST等)和其它简单的功能。

APIView

APIView是drf中所有view的父类,本身继承于Django的VIew,只有简单的调度方法和健壮检查。

和View的不同

  • 请求和返回使用的drf的Request Response而不是django的HttpRequest HttpResponse
  • 请求传入时进行身份验证,并在传给处理方法前进行权限检验。
  • 任何APIException都会被捕捉并放入合适的想要中。

使用

response返回的内容需是序列化的json。

先创建在goods包下新建serializer.py

某字段结果可能有多条时,记得加many=True!否则无法正确显示。

class GoodsSerializer(serializers.ModelSerializer):
images = GoodImagesSerializer(many=True) class Meta:
model = Goods
fields = "__all__"

views

因为会有多条商品信息,所以这里也要记得加many=True!否则会报错。

class GoodsListView(APIView):
# 重写get方法 进行get访问时会进行的处理
def get(self, request):
goods = Goods.objects.all()[:10]
goods_list = GoodsSerializer(goods, many=True)
return Response(goods_list.data)

在url下配置view的访问,别忘了这是基于类的视图,所以要加上as_view()方法。

url(r'^GoodsListView/', GoodsListView.as_view()),

GenericAPIView

GenericAPIView继承于APIView,为标准list和detail详情提供了常用行为,每个GenericAPIView都会和一个或多个mixin联合使用。

属性

  • 基础设置

    • queryset 需要返回的结果集
    • serializer_class 用于序列化的serializer
    • lookup_field 查找单个model实例时的字段,默认为pk(主键)
    • lookup_url_kwarg
  • 分页
    • pagination_class
  • 过滤
    • filter_backend

使用

先在views里设置个分页

class StandardResultsSetPagination(PageNumberPagination):
page_size = 10
page_size_query_param = 'page_size' # GoodsListView/?p=2&page_size=3
page_query_param = 'p' # GoodsListView/?p=2
max_page_size = 100
class GoodsListView(GenericAPIView, ListModelMixin):
"""
商品列表
"""
queryset = Goods.objects.all()
# 序列化
serializer_class = GoodsSerializer
# 分页
pagination_class = StandardResultsSetPagination
# 过滤 /排序、搜索
filter_backends = (filters.DjangoFilterBackend, SearchFilter, OrderingFilter)
ordering_fields = ('add_time', )
search_fields = ('name', 'goods_desc') def get(self, request, *args, **kwargs):
return self.list(request, *args, **kwargs)

ListAPIView

除了继承GenericAPIView加ListModelMixin,重写了下get方法,没干别的事。

源码

class ListAPIView(mixins.ListModelMixin,
GenericAPIView):
"""
Concrete view for listing a queryset.
"""
def get(self, request, *args, **kwargs):
return self.list(request, *args, **kwargs)

使用

class GoodsListView(ListAPIView):
"""
商品列表
"""
queryset = Goods.objects.all()
# 序列化
serializer_class = GoodsSerializer
# 分页
pagination_class = StandardResultsSetPagination
# 过滤 /排序、搜索
filter_backends = (filters.DjangoFilterBackend, SearchFilter, OrderingFilter)
ordering_fields = ('add_time', )
search_fields = ('name', 'goods_desc')

作用就是少写继承,不用重写get...

类似有:

class CreateAPIView(mixins.CreateModelMixin,
GenericAPIView)
class RetrieveAPIView(mixins.RetrieveModelMixin,
GenericAPIView)
class RetrieveUpdateAPIView(mixins.RetrieveModelMixin,
mixins.UpdateModelMixin,
GenericAPIView)

等等常用到的行为都被封装了。

GenericViewSet

继承于GenericAPIView,并提供默认的get_object,get_queryset方法和其他通用视图基本行为,但默认情况下不包含任何操作。为了使用GenericViewSet类,您将覆盖该类并混合所需的mixin类,或者显式定义动作实现。

源码

class GenericViewSet(ViewSetMixin, generics.GenericAPIView):
pass

只是比GenericAPIView多了ViewSetMixin。

在ViewSetMixin里重写了as_view方法,可以更方便的绑定actions。

使用

class GoodsListViewSet(GenericViewSet, ListModelMixin):
"""
商品列表
"""
queryset = Goods.objects.all()
# 序列化
serializer_class = GoodsSerializer
# 分页
pagination_class = StandardResultsSetPagination

urls配置

将get请求 绑定到list() action

goods_list = GoodsListViewSet.as_view({
'get': 'list'
}) urlpatterns = [
....
url(r'^goodsList/', goods_list),
]

使用router

VIewSet常常和router使用,router可以自动将常用的 get绑定list,post绑定create这些操作完成。

urls中设置如下,上面的那两个设置就不需要了。

from rest_framework.routers import DefaultRouter

router = DefaultRouter()
router.register('goodsList', GoodsListViewSet) urlpatterns = [
.....
url(r'^', include(router.urls)),
]

ReadOnlyModelViewSet

继承于GenericViewSet,混用RetrieveModelMixin,ListModelMixin。这样我们就既可以用它来访问列表,也可以用来访问详情。

源码

class ReadOnlyModelViewSet(mixins.RetrieveModelMixin,
mixins.ListModelMixin,
GenericViewSet):
"""
A viewset that provides default `list()` and `retrieve()` actions.
"""
pass

使用

class GoodsViewSet(viewsets.ReadOnlyModelViewSet):
"""
商品列表
商品详情
"""
queryset = Goods.objects.all()
# 序列化
serializer_class = GoodsSerializer
# 分页
pagination_class = StandardResultsSetPagination

绑定router

router.register('goods', GoodsViewSet)

访问接口

drf中的各种view,viewset的更多相关文章

  1. DRF中的APIView、GenericAPIView、ViewSet

    1.APIView(rest_framework.views import APIView),是REST framework提供的所有视图的基类,继承自Django的View. 传入到视图方法中的是R ...

  2. drf中View和router的详解

    Rest Framework 视图和路由 因为涉及到视图层面了,而且下面的例子会反复用到request.data,所以我决定带大家稍微看下源码,感兴趣的可以自己深入了解 无论是View还是APIVie ...

  3. DRF中的视图集的使用

    1.说明:DRF框架中的视图集: 在drf开发接口中,使用GenericAPIView和视图扩展类结合起来完成接口功能是一件很常见的事情,所以,drf的作者帮我们提前把  GenericAPIView ...

  4. DRF中的APIView源码分析

    首先写一个简单的drf接口 from rest_framework.views import APIView from rest_framework.response import Response ...

  5. day71:drf:API接口&Restful API规范&Django Rest Framework&drf中的序列化和反序列化功能

    目录 1.web应用模式 2.API接口 3.Restful API规范 4.序列化 5.Django Rest Framework 1.drf的简单介绍 2.drf的特点 3.如何安装drf 4.d ...

  6. 如何在Spring MVC Test中避免”Circular view path” 异常

    1. 问题的现象 比如在webConfig中定义了一个viewResolver public class WebConfig extends WebMvcConfigurerAdapter { //配 ...

  7. ASP.NET MVC3中Controller与View之间的数据传递总结

    一.  Controller向View传递数据 1.       使用ViewData传递数据 我们在Controller中定义如下: ViewData["Message_ViewData& ...

  8. ASP.NET MVC3中Controller与View之间的数据传递

    在ASP.NET MVC中,经常会在Controller与View之间传递数据,因此,熟练.灵活的掌握这两层之间的数据传递方法就非常重要.本文从两个方面进行探讨: 一.  Controller向Vie ...

  9. iOS 在UITableViewCell中加入自定义view时view的frame设定注意

    由于需要重用同一个布局,于是在cellForRowAtIndexPath中把自定义view加在了cell上,我是这样设定view的frame的 var screenFrame = UIScreen.m ...

随机推荐

  1. shell 查找字符串中字符出现的位置

    #!/bin/bash a="The cat sat on the mat" test="cat" awk -v a="$a" -v b=& ...

  2. 条件随机场CRF原理介绍 以及Keras实现

    本文是对CRF基本原理的一个简明的介绍.当然,“简明”是相对而言中,要想真的弄清楚CRF,免不了要提及一些公式,如果只关心调用的读者,可以直接移到文末. 图示# 按照之前的思路,我们依旧来对比一下普通 ...

  3. <每日 1 OJ> -Table

    上图是一个Mysql查询结果图,我们看到这个表格非常漂亮,只需要使用”+”和”-”两个符号就可以打印,现在你的任务是打印一个n×m的表格我们定义单位长度(水平方向有三个”-”,竖直方向有一个”| ”, ...

  4. 每天一个命令-cp 命令

    cp命令用来将一个或多个源文件或者目录复制到指定的目的文件或目录.它可以将单个源文件复制成一个指定文件名的具体的文件或一个已经存在的目录下.cp命令还支持同时复制多个文件,当一次复制多个文件时,目标文 ...

  5. 一个按权重(weight)进行LB的算法

    package netty; import com.google.common.collect.ImmutableList; import lombok.SneakyThrows; import ja ...

  6. 解决git 出现 Your account has been blocked问题

    使用git 出现 Your account has been blocked 无法从远程pull代码下来, 解决方案如下: $ git push origin masterGitLab: Your a ...

  7. Ubuntu 安装最新版 (1.12) Golang 并使用 go mod

    wget https://dl.google.com/go/go1.12.4.linux-amd64.tar.gz sudo tar -zxvf go1.12.4.linux-amd64.tar.gz ...

  8. java base64编码解码

    第一种.DatatypeConverter 使用JDK自带DatatypeConverter.java类实现,JDK版本必须>=1.6 /** * @Description base64编码 * ...

  9. vue-cli webpack打包开启Gzip 报错—— Cannot find module 'compression-webpack-plugin

    异常描述: 复用以前框架,打包的时候报异常提示: Cannot find module 'compression-webpack-plugin" 然后安装插件: npm install -- ...

  10. 使用evenlet包实现 concurrent.futures.executor包的鸭子类

    适配成同一个同样的公有方法. # -*- coding: utf-8 -*- # @Author : ydf # @Time : 2019/7/3 10:35 import time import w ...