首页 > 其他分享 >drf之视图集

drf之视图集

时间:2022-09-30 19:45:12浏览次数:45  
标签:ser return 之视 get self request class 图集 drf


一、试图基类

# 我们之前学习的drf在编写视图类的时候 是继承了APIView 和 GenericAPIView 

# GenericAPIView的属性和方法回顾:
    -属性
        -queryset    # 要序列化的数据
        -serializer_class # 序列化类
        -lookup_field  # 通过get_object  获取单个对象的查询key值,value值是路由中传进来的
        -filter_backends  # 过滤类
        -pagination_class # 分页类
    -方法
         -get_queryset # 获取要序列化的数据
        -get_object   # 根据lookup_field配置的参数获取单个对象
        -get_serializer # 获取序列化类,咱们直接用的
        
        -get_serializer_class # 获取序列化类,不是咱们直接用的get_serializer调用了它
        -filter_queryset # 跟过滤有关系
        -paginate_xxx    # 跟分页有关

1.1基于APIView写的五个接口

class UserView(APIView):
    def get(self, request):
        res_list = User.objects.all()
        ser = UserSerializer(instance=res_list, many=True)
        return Response(ser.data)

    def post(self, request):
        ser = UserSerializer(data=request.data)
        if ser.is_valid():
            ser.save()
            return Response({'code': 100, 'msg': "新增成功"}, status=201)
        else:
            return Response({'code': 101, 'msg': ser.errors})


class UserDetailView(APIView):
    def get(self, request, pk):
        obj = User.objects.filter(pk=pk).first()
        ser = UserSerializer(instance=obj)
        return Response(ser.data)

    def put(self, request, pk):
        obj = User.objects.filter(pk=pk).first()
        ser = UserSerializer(instance=obj, data=request.data)
        if ser.is_valid():
            ser.save()
            return Response({'code': 100, 'msg': "修改成功"}, status=201)
        else:
            return Response({'code': 101, 'msg': ser.errors})

    def delete(self, request, pk):
        User.objects.filter(pk=pk).delete()
        return Response('')

# 这个时候我们发现 有许多代码重复了  就可以继承GenericAPIView来解决

2.使用GenericAPIView来编写五个接口

class UserView(GenericAPIView):
    # 配置两个 类属性
    queryset = User.objects.all()
    serializer_class = UserSerializer

def get(self, request): res_list = self.get_queryset() # 提高扩展性 ser = self.get_serializer(instance=res_list, many=True) return Response(ser.data) def post(self, request): ser = self.get_serializer(data=request.data) if ser.is_valid(): ser.save() return Response({'code': 100, 'msg': "新增成功"}, status=201) else: return Response({'code': 101, 'msg': ser.errors}) class UserDetailView(GenericAPIView): queryset = User.objects.all() serializer_class = UserSerializer def get(self, request, pk): # obj = self.get_queryset().filter(pk=pk).first() # queryset.get({'pk':'有名分组分出来的'} obj = self.get_object() # 根据传入的pk,获取一条数据 ser = self.get_serializer(instance=obj) return Response(ser.data) def put(self, request, pk): obj = self.get_object() ser = self.get_serializer(instance=obj, data=request.data) if ser.is_valid(): ser.save() return Response({'code': 100, 'msg': "修改成功"}, status=201) else: return Response({'code': 101, 'msg': ser.errors}) def delete(self, request, pk): self.get_queryset().filter(pk=pk).delete() return Response('') # 我们可以把类中方法给单独写一个类 然后在视图类中继承即可 然后drf中有写好的五个试图扩展了 我们只需导入 继承即可

二、五个试图扩展类

# 必须配合GenericAPIView使用,不能配合APIView使用
from rest_framework.mixins import RetrieveModelMixin, CreateModelMixin, UpdateModelMixin, DestroyModelMixin, \
    ListModelMixin


# CreateModelMixin:写了一个Create方法,就是原来咱们post 新增一个
# RetrieveModelMixin:写了一个retrieve,就是咱们原来的get  获取一个
# UpdateModelMixin:写了一个update方法,就是咱们原来的put  修改
# ListModelMixin:写了一个list方法,就是原来咱们的get  获取全部
# DestroyModelMixin:写了一个destroy方法,就是原来咱们的delete 删除一个

1.代码演示

class UserView(GenericAPIView, ListModelMixin, CreateModelMixin):
    # 配置两个 类属性
    queryset = User.objects.all()
    serializer_class = UserSerializer

    def get(self, request, *args, **kwargs):
        return self.list(request, *args, **kwargs)

    def post(self, request, *args, **kwargs):
        return self.create(request, *args, **kwargs)


class UserDetailView(GenericAPIView, RetrieveModelMixin, UpdateModelMixin, DestroyModelMixin):
    queryset = User.objects.all()
    serializer_class = UserSerializer

    def get(self,request, *args, **kwargs):
        return self.retrieve(request, *args, **kwargs)

    def put(self, request, *args,**kwargs):
        return self.update(request,*args,**kwargs)

    def delete(self, request, *args,**kwargs):
        return self.destroy(request,*args,**kwargs)

# 因为继承的父类中写了对应的方法 所以我们在视图类中可以不用在写对应的方法了

三、九个试图子类

from rest_framework.generics import ListAPIView,CreateAPIView,   RetrieveAPIView,DestroyAPIView,UpdateAPIView
from rest_framework.generics import ListCreateAPIView,     RetrieveUpdateDestroyAPIView,RetrieveUpdateAPIView,RetrieveDestroyAPIView

ListAPIView  # 获取全部
CreateAPIView  # 新增一个
RetrieveAPIView  # 获取一个
DestroyAPIView  # 删除一个
UpdateAPIView  # 修改

ListCreateAPIView  # 获取全部和新增一个

RetrieveUpdateAPIView  # 获取一个和修改
RetrieveDestroyAPIView  #获取一个和删除一个
RetrieveUpdateDestroyAPIView  # 获取一个和修改一个和删除一个

1.代码演示

# 正常来讲  Destroy+Update 应该有一个  ,作者没加
class UserView(ListCreateAPIView):
    # 配置两个 类属性
    queryset = User.objects.all()
    serializer_class = UserSerializer
class UserDetailView(RetrieveUpdateDestroyAPIView):
    queryset = User.objects.all()
    serializer_class = UserSerializer


# 然后我们发现这个还是可以优化一下 所以drf也有这个写好的类 我们只需要导入继承即可

四、视图集

# 视图类
from rest_framework.viewsets import ModelViewSet,ReadOnlyModelViewSet
class UserView(ModelViewSet):
    queryset = User.objects.all()
    serializer_class = UserSerializer

# ModelViewSet 是有正常的五个接口 都有 只需要继承它 那么五个的功能就会都有了
# ReadOnlyModelViewSet  只有获取一个和获取全部

# 我们继承了ModelViewSet 或则 ReadOnlyModelViewSet 那么需要改变路由的编写 如果还是之前的路由 那么就会直接报错  所以需要修改路由

# 路由
path('user/', views.UserView.as_view({'get': 'list', 'post': 'create'})),
path('user/<int:pk>', views.UserView.as_view({'get': 'retrieve', 'put': 'update', 'delete': 'destroy'})),  # 需要这样写才能不会报错

1.ModelViewSet的源码分析

# 我们现在看一下ModelViewSet的源码为什么路由需要这样写才可以不会报错

# 那么源码的入口还是从as_view入手

# 然后我们写的视图类中没有as_view的方法 所以去继承的ModelViewSet中查找 我们可以看到ModelViewSet中没有写方法 所以取它继承的类中查找
class ModelViewSet(mixins.CreateModelMixin,
                   mixins.RetrieveModelMixin,
                   mixins.UpdateModelMixin,
                   mixins.DestroyModelMixin,
                   mixins.ListModelMixin,
                   GenericViewSet):
    pass
# 然后根据名字的查找顺序 从左到右 我们可以知道前面的五个也是没有as_view的方法的 
# 然后我们就去到了GenericViewSet类中查找 我们可以看到GenericViewSet类中也没有写什么方法 所以去它继承的类中查找
class GenericViewSet(ViewSetMixin, generics.GenericAPIView):
    pass
# 所以我们需要先去ViewSetMixin中查找as_view方法 如果有就不用找后面的类了  我们可以看到是有这个方法的

2.ViewSetMixin的源码分析

    @classonlymethod
    def as_view(cls, actions=None, **initkwargs):  # actions就是我们在路由中匹配的字典中的参数
        if not actions:  # 所以如果我们不参数就会抛出异常 所以这里规定我们一定要传参数  eg:{'get': 'list', 'post': 'create'}
            raise TypeError("The `actions` argument must be provided when "
                            "calling `.as_view()` on a ViewSet. For example "
                            "`.as_view({'get': 'list'})`"
        def view(request, *args, **kwargs):  # 然后执行view函数
            for method, action in actions.items():  # actions.items就是把我们传得字典的K:V
            # 然后通过for循环把我们传得字典中一个一个循环出来 然后通过K赋值给method  V赋值给action
            # 所以现在method现在就是'get'  action就是'list'
                handler = getattr(self, action)  
                # 然后通过反射getattr获取self中的list方法 self就是我们写的视图类
                # 因为我们视图类继承了ModelViewSet 然后这个类有继承了ListModelMixin 然后在这个类中有一个list方法 
                # 所以我们自己写的试图类 也可以使用这个list方法 所以现在handler就是这个list方法
                setattr(self, method, handler)
                # 然后在通过反射setattr 把list这个方法设置到我们自己写的试图类中  self.method=handler 就是等于 self.get=list
            return self.dispatch(request, *args, **kwargs)
        return csrf_exempt(view)

# 所以现在我们只要继承了ModelViewSet或则ReadOnlyModelViewSet 路由都需要变成views.UserView.as_view({'get': 'list', 'post': 'create'}))这样写

# 但是我们视图类中的方法名可以不需要在写post、get、put...
# 可以写我们自己想要的方法名 因为会先从对象本身开始找 所以不会报错

 

标签:ser,return,之视,get,self,request,class,图集,drf
From: https://www.cnblogs.com/stephenwzh/p/16745935.html

相关文章

  • 【Django-rest-framework框架】第04回 视图集
    目录1.两个视图基类1.1GenericAPIview属性和方法1.2基于APIView写5个接口1.3基于GenericAPIview写5个接口2.5个视图扩展类3.9个视图子类4.视图集5.源码分析ViewSet......
  • drf学习笔记
    今日内容概要两个视图基类五个视图扩展类九个视图子类视图集今日内容详细两个视图基类补充:GenericAPIView:属性:1.queryset#要序列化的数据......
  • drf-04
    目录请求与响应请求Request响应Response视图组件视图的继承2个视图基类drf内容框架作业请求与响应请求Request1.request:drf中的request不再是django默认的HttpRequest对......
  • drf之视图组件
    drf请求与响应1.request请求:通过http请求,经过OSI协议,拿着前端提供的数据给了django,django把请求给了request的对象,drf又对request进行了一次封装,每次请求都是一个新的req......
  • drf学习-4
    一、drf入门流程前后端开发模式API接口postman使用序列化和反序列化restful规范drf:第三方app—快速实现符合restful规范的接口以后写的都是视图类都是继承APIView......
  • drf请求与相应(Request,Response),drf能够解析的请求编码,响应编码,GenericAPIView和APIVi
    drf请求与响应Request类(请求)Response类(响应)drf 能够解析的请求编码,响应编码能够解析的请求编码响应编码GenericAPIView和APIView(2个视图基......
  • drf学习笔记
    今日内容概要drf之请求与响应drf之视图组件两个视图基类今日内容详细补充知识反射:通过字符串动态的获取,设置,判断对象中得属性或方法-getattr:res=getattr(se......
  • drf之试图基类
    一、请求与响应#现在我们在写视图类的时候都是继承了APIView#而APIView的request都变成了新的request了跟继承了View的request已经不一样了#所以每次请求都是一个......
  • drf之请求与响应,drf之视图组件,2个视图基类
    1.drf之请求与响应在继承drf中的APIView时Request它的请求对象request就不是原来的那个request了(具体去看APIView源码分析),所以没钱请求的request都是一个新的对象,这......
  • drf之请求与响应、drf之视图组件、2个视图基类
    drf之请求与响应Request和Response类继承APIView后,请求对象:request,每一次请求都是一个新的requestRequest类:属性或方法data:POST、PUT、PATCH请求方式解析后的数据......