zoukankan      html  css  js  c++  java
  • Django通用视图APIView和视图集ViewSet的介绍和使用(Django编程-1)

    1.APIView

    DRF框架的视图的基类是 APIView
    APIView的基本使用和View类似

    1. Django默认的View请求对象是 HttpRequest,REST framework 的请求对象是 Request。
      Request对象的数据是自动根据前端发送数据的格式进行解析之后的结果。
      HttpRequest.GET ————> Request.query_params
      HttpRequest.POST 、HttpRequest.body————> Request.data

    2. Django默认的View响应对象是 HttpResponse(以及子类),REST framework 的响应对象是Response。
      构造方式:
      Response(data, status=None, template_name=None, headers=None, content_type=None)

      参数说明:

      • List item

      • data: 为响应准备的序列化处理后的数据;

      • status: 状态码,默认200;

      • template_name: 模板名称,如果使用HTMLRenderer时需指明;

      • headers: 用于存放响应头信息的字典;

      • content_type: 响应数据的Content-Type,通常此参数无需传递,REST framework会根据前端所需类型数据来设置该参数。

    3. 支持定义的属性:

      • authentication_classes列表或元祖,身份认证类
      • permissoin_classes列表或元祖,权限检查类
      • throttle_classes列表或元祖,流量控制类。
    4. 在APIView中仍以常规的类视图定义方法来实现get() 、post() 或者其他请求方式的方法。如下:

      
          ''' serializers.py '''
      
          class BookSerializer(serializers.ModelSerializer):
              
              class Meta:
                  model = BookInfo    # 设置关联模型     model就是关联模型
                  fields = '__all__'  # fields设置字段   __all__表示所有字段
                  
          '''urls.py'''
          url(r'^center/$',views.CenterView.as_view())
      
      
      
          ''' views.py '''
          class CenterView(APIView):
      
              def get(self,request):
      
                  #  以前的 HttpRequest.GET  
                  #  现在的
                  # /center/?a=100&b=python
                  params = request.query_params
                  print(params)
      
                  # 响应不同的第一个
                  dict = {
                      'name':'hello'
                  }
                  # return JsonResponse(dict)
      
                  return Response(dict)
      
                  # return HttpResponse('get')
      
              def post(self,request):
      
                  # 以前的 HttpRequest.POST,HttpRequest.body
                  # 现在
                  # form 表单提交数据
                  data = request.data
                  print(data)
                  
                  return HttpResponse('post')
          
      

    2. 实例:使用APIView实现列表功能

    
        '''urls.py'''
        url(r'^books/$',views.BookListAPIView.as_view())
    
        '''views.py'''
    
        class BookListAPIView(APIView):
            '''书籍列表页'''
            def get(self,request):
                # 1.获取所有书籍
                books = BookInfo.objects.all()
                # 2.通过序列化器的转换(模型转换为JSON)
                serializer = BookSerializer(book,many=True)
                # 3.返回响应
                return Response(serializer.data)
    
            def post(self,request):
    
                # 1.接收参数
                data = request.data
                # 2.验证参数(序列化器的校验)
                serializer = BookSerializer(data=data)
                serializer.is_valid(raise_exception=True)
                # 3.数据入库
                serializer.save()
                # 4.返回响应
                return Response(serializer.data)
    

    3.GenericAPIView

    1. GenericAPIView是继承自APIView,GenericAPIView肯定在APIView的基础上 封装了一些属性和方法:增加了对于列表视图和详情视图可能用到的通用方法和属性的支持

      属性:
      queryset 设置结果集
      serializer_class 设置序列化器
      lookup_field 查询指定的对象

      方法:
      get_queryset(self) 返回视图使用的查询集
      get_serializer(self,_args, *_kwargs) 返回序列化器对象
      get_object(self) 返回详情视图所需的模型类数据对象

    2. 通常使用时,可搭配一个或多个扩展类(Mixin类,详见4.)

    
        ##########GenericAPIView列表视图##################
    
        '''urls.py'''
        url(r'^books/$',views.BookListGenericAPIView.as_view())
    
    
    
        '''views.py'''
    
        class BookListGenericAPIView(GenericAPIView):
            
            '''列表视图'''
    
            # 查询结果集
            queryset = BookInfo.objects.all()
            # 序列化器类
            serializer_class = BookSerializer
    
            def get(self,request):
    
                # 1.获取所有书籍
                # books = BookInfo.objects.all()
                # 上面写法也可以,但属性就白白浪费了,没有充分发挥属性的作用
    
                books = self.get_queryset()
    
                # 2.创建序列化器
                # serializer = BookSerializer(books,many=True)
                # 以上写法也可可以,但是还是没有发挥属性的作用
    
                # get_serializer()相当于BookSerializer()
                serializer = self.get_serializer(book,many=True)
    
                # 3.返回响应
                return Response(serializer.data)
    
                
            def post(self,request):
    
                # 1.获取参数
                data = request.data
                # 2.创建序列化器
                serializer = self.get_serializer(data=data)
                # 3.校验
                serializer.is_valid(raise_exception=True)
                # 4.保存
                serializer.save()
                # 5.返回响应
                return Response(serializer.data)
    
    	class BookDetailGenericAPIView(GenericAPIView):
    
            '''详情视图'''
    
            # 查询结果集
            queryset = BookInfo.objects.all()
            # 序列化器类
            serializer_class = BookSerializer
    
            # 默认是pk   修改后以下参数都要变
            lookup_field = 'id'
    
            def get(self,request,id):
    
                # 1.获取对象
                book = self.get_object()
                # 2.创建序列化器
                serializer = self.get_serializer(book)
                # 3.返回响应
                return Response(serializer.data)
    
    
            def put(self,request,id):
    
                # 1.获取对象
                book = self.get_object()
                # 2.接收参数
                data = request.data
                # 3.创建序列化器
                serializer = self.get_serializer(instance=book,data=data)
                # 4.验证
                serializer.is_valid(raise_exception=True)
                # 5.保存(更新)
                serializer.save()
                # 3.返回响应
                return Response(serializer.data)
    
    
    
            def delete(self,request,pk):
                # 1.获取对象
                book = self.get_object()
                # 2.删除
                book.delete()
                # 3.返回响应
                return Response(status=status.HTTP_204_NO_CONTENT)
    
    1. GenericAPIView和Mixin配合使用
      mixin类提供用于提供基本视图行为的操作。请注意,mixin类提供了操作方法,而不是直接定义处理程序方法,例如.get()和.post()。这允许更灵活的行为组合。

      • ListModelMixin
        提供一种.list(request, *args, **kwargs)实现列出查询集的方法。

      • CreateModelMixin
        提供.create(request, *args, **kwargs)实现创建和保存新模型实例的方法。

      • RetrieveModelMixin
        提供一种.retrieve(request, *args, **kwargs)方法

      • UpdateModelMixin
        提供.update(request, *args, **kwargs)实现更新和保存现有模型实例的方法。

      • DestroyModelMixin
        提供一种.destroy(request, *args, **kwargs)实现删除现有模型实例的方法。

    
        '''urls.py'''
     	url(r'^booklist/$',views.BookListGenericMixinAPIView.as_view())
    
    
        '''views.py'''
    
        # ListModelMixin        获取全部对象(列表)
        # CreateModelMixin      新增资源
        # RetrieveModelMixin    获取一个资源
        # UpdateModelMixin      更新一个资源
        # DestoryModelMixin     删除一个资源
    
        class BookListGenericMixinAPIView(ListModelMixin,CreateModelMixin,GenericAPIView):
    
           # 查询结果集
            queryset = BookInfo.objects.all()
            # 序列化器类
            serializer_class = BookSerializer
    
            def get(self,request):
    
                return self.list(request)
    
            def post(self,request):
    
                return self.create(request)
    
            def post(self,request):
    
                return self.create(request)
            
        class BookDetailGenericMixinAPIView(R,U,D):
    
            def get(self,request):
    
                return self.retrieve(request)
    
            def put(self,request):
    
                return self.update(request)
    
            def delete(self,request):
    
                return self.destroy(request)
    
    

    6.三级视图

    • CreateAPIView
      提供post方法处理程序。

    • ListAPIView
      用于只读端点以表示模型实例的集合。
      提供get方法处理程序。

    • RetrieveAPIView
      用于表示单个模型实例的只读端点。
      提供get方法处理程序。

    • DestroyAPIView
      用于单个模型实例的仅删除端点。
      提供delete方法处理程序。

    • UpdateAPIView
      用于单个模型实例的仅更新端点。
      提供put和patch方法处理程序。

    • ListCreateAPIView
      用于读写端点以表示模型实例的集合。
      提供get和post方法处理程序。

    • RetrieveUpdateAPIView
      用于读取或更新端点以表示单个模型实例。
      提供get,put并且patch方法处理。

    • RetrieveDestroyAPIView
      用于读取或删除端点以表示单个模型实例。
      提供get和delete方法处理程序。

    • RetrieveUpdateDestroyAPIView
      用于读写 - 删除端点以表示单个模型实例。
      提供get,put,patch和delete方法处理。

    
        '''urls.py'''
        url(r'^booklist/$',views.Book2ListAPIView.as_view())
    
    
        '''views.py'''
    
        class Book2ListAPIView(ListAPIView):
    
            # 查询结果集
            queryset = BookInfo.objects.all()
            # 序列化器类
            serializer_class = BookSerializer
    
    

    7.视图集Viewset

    1. APIView,GenericAPIView,ListAPIView 都是继承自View
      继承自View的类视图,只能定义相同的一个函数名,例如:只能定义一个get,post方法

    2. 列表视图中 设置了 queryset,serializer_class get,post
      详情视图中也设置了 queryset,serializer_class get,put,delete
      能否将这两个视图合并???
      我们是可以将 列表和详情视图 组合到一起的,称之为 视图集 ViewSet
      它不提供任何方法处理程序( 如get(),post() ),而是提供了诸如list() create()之类的操作

    
        class BookViewSet(ViewSet):
    
            # get
            def list(self,request):
                queryset = BookInfo.objects.all()
                serializer = BookInfoSerializer(queryset,many=True)
                return Response(serializer.data)
            # get
            def retrieve(self,request,pk=None):
                queryset = BookInfo.objects.all()
                user = get_object_or_404(queryset, pk=pk)
                serializer = BookInfoSerializer(user)
                return Response(serializer.data)
    
    
        '''urls.py'''
        # 继承自 ViewSet的url 可以自动生成
        from rest_framework.routers import DefaultRouter,SimpleRouter
    
        # Router:的相同点  都是可以自动生成url
        # Router:的不同点
            #       DefaultRouter    可以在根路由下显示
            #       SimpleRouter     不可以在根路由下显示,且会报404
        
       
        # 1.创建router对象
        router = DefaultRouter()
    
        # 2.设置正则
        # router的原理
        # router 会自动生成两个url   一个是列表视图的url  另一个是详情视图的url
    
        # 参数1:正则,  只需要设置列表视图和详情视图公共的部分
            # 例如:  booklist/是列表视图
                    #   booklist/id 是详情视图
                    # 公共部分是 booklist  不包括/
        # 参数2:视图集
        # 参数3:base_name  只是我们名字url name的一个前缀
                # 例如:列表视图的名字: base_name-list     book-list
                #       详情视图的名字: base_name-detail    book-detail
        router.register(r'booklist',views.BookModelViewset,base_name='book')
    
        # 3.将自动生成的url  添加到 urlpatterns中
        # router.urls  urls 这个属性 存放了自动生成的url
        urlpatterns +=  router.urls
    
    
    
    
        # ModelViewSet 其实就是继承自 GenericAPIView,同时继承了5个扩展
        class BookModelViewset(ModelViewSet):
    
            queryset = BookInfo.objects.all()
            serializer_class = BookSerializer
    
    
  • 相关阅读:
    【HDOJ】4982 Goffi and Squary Partition
    【HDOJ】4983 Goffi and GCD
    【算法导论】学习笔记——第7章 快速排序
    【算法导论】学习笔记——第6章 堆排序
    【HDOJ】4956 Poor Hanamichi
    【HDOJ】2492 Ping pong
    【Linux】鸟哥的Linux私房菜基础学习篇整理(十二)
    【Linux】鸟哥的Linux私房菜基础学习篇整理(十一)
    统计硬币
    放大的X
  • 原文地址:https://www.cnblogs.com/kadycui/p/10289396.html
Copyright © 2011-2022 走看看