1 2个视图基类-2 5个视图扩展类-3 9个视图子类-视图集-路由组件-补充点
1 2个视图基类
# django 内置的View # drf 的APIView ,继承自View # GenericAPIView -两个重要的类属性: queryset = Book.objects.all() # 要序列化的数据 serializer_class = BookSerializer # 序列化类 -几个重要的方法: self.get_object() self.get_queryset() self.get_serializer(instance=book_list, many=True)
1.1使用APIView写五个接口
class BookView(APIView): def get(self,request): book_list=Book.objects.all() # 如果写了many=True,ser是哪个类的对象? ser=BookSerializer(instance=book_list,many=True) print(type(ser)) # ListSerializer的对象---》[BookSerializer,BookSerializer,BookSerializer] return Response(ser.data) def post(self,request): ser=BookSerializer(data=request.data) if ser.is_valid(): ser.save() return Response({'code':'100','msg':'创建成功'}) else: return Response({'code': '101', 'msg': '创建失败','err':ser.errors}) class BookDetailView(APIView): def get(self,request,*args,**kwargs): # pk 从kwargs中去 book=Book.objects.filter(pk=kwargs['pk']).first() # 如果写了many=True,ser是哪个类的对象? ser=BookSerializer(instance=book) print(type(ser)) # BookSerializer的对象 return Response(ser.data) #put:全局修改 patch:局部修改 def put(self,request,*args,**kwargs): # pk 从kwargs中去 book=Book.objects.filter(pk=kwargs['pk']).first() ser=BookSerializer(instance=book,data=request.data) if ser.is_valid(): ser.save() return Response({'code': '100', 'msg': '创建成功'}) else: return Response({'code': '101', 'msg': '创建失败', 'err': ser.errors})
1.2继承GenericAPIView写5个接口
class BookView(GenericAPIView): queryset = Book.objects.all() # 要序列化的数据 serializer_class = BookSerializer # 序列化类 def get(self, request): # print('dsafdasdfasd') book_list = self.get_queryset() # 获取要序列化的数据,不要直接使用self.queryset,而要用self.get_queryset() # 获取序列化类,要使用self.get_serializer ser = self.get_serializer(instance=book_list, many=True) print(type(ser)) # ListSerializer的对象---》[BookSerializer,BookSerializer,BookSerializer] 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': '创建成功'}) else: return Response({'code': '101', 'msg': '创建失败', 'err': ser.errors}) class BookDetailView(GenericAPIView): queryset = Book.objects.all() # 要序列化的数据 serializer_class = BookSerializer # 序列化类 #传入的必须叫pk,否则,get_object就拿不到对象 def get(self, request, *args, **kwargs): # 获取单个对象, book = self.get_object() # 如果写了many=True,ser是哪个类的对象? ser = self.get_serializer(instance=book) print(type(ser)) # BookSerializer的对象 return Response(ser.data) # put:全局修改 patch:局部修改 def put(self, request, *args, **kwargs): # pk 从kwargs中去 book = self.get_object() ser = self.get_serializer(instance=book, data=request.data) if ser.is_valid(): ser.save() return Response({'code': '100', 'msg': '创建成功'}) else: return Response({'code': '101', 'msg': '创建失败', 'err': ser.errors})
2 5个视图扩展类
# 视图扩展类---》不是视图类---》没有继承视图类(View,APIView...子类) ListModelMixin, #获取所有 CreateModelMixin,#创建一个 RetrieveModelMixin,#查询一个 UpdateModelMixin,#更新一个 DestroyModelMixin#删除一个
# class ListModeMixin(): # def list(self, request): # print('dsafdasdfasd') # book_list = self.get_queryset() # 获取要序列化的数据,不要直接使用self.queryset,而要用self.get_queryset() # # # 获取序列化类,要使用self.get_serializer # ser = self.get_serializer(instance=book_list, many=True) # print(type(ser)) # ListSerializer的对象---》[BookSerializer,BookSerializer,BookSerializer] # return Response(ser.data) from rest_framework.mixins import ListModelMixin,CreateModelMixin,RetrieveModelMixin,UpdateModelMixin,DestroyModelMixin class BookView(GenericAPIView,ListModelMixin,CreateModelMixin): queryset = Book.objects.all() # 要序列化的数据 serializer_class = BookSerializer # 序列化类 def get(self,request): print('获取所有') return super().list(request) def post(self, request): return super().create(request) class BookDetailView(GenericAPIView,RetrieveModelMixin,UpdateModelMixin,DestroyModelMixin): queryset = Book.objects.all() # 要序列化的数据 serializer_class = BookSerializer # 序列化类 #传入的必须叫pk,否则,get_object就拿不到对象 def get(self, request, *args, **kwargs): print('获取一') return super().retrieve(request, *args, **kwargs) # put:全局修改 patch:局部修改 def put(self, request, *args, **kwargs): # pk 从kwargs中去 return super().update(request, *args, **kwargs) def delete(self,request, *args, **kwargs): return super().destroy(request, *args, **kwargs)
3 9个视图子类
# 获取所有,新增一个,获取所有和新建一个 ListAPIView,CreateAPIView,ListCreateAPIView # 获取单个,更新一个,删除一个 RetrieveAPIView,UpdateAPIView,DestroyAPIView # RetrieveUpdateDestroyAPIView,RetrieveUpdateAPIView,RetrieveDestroyAPIView # 继承9个视图子类后,只需要在视图类中写两个类属性即可
class BookView(ListCreateAPIView): # 获取所有和新增一个 queryset = Book.objects.all() # 要序列化的数据 serializer_class = BookSerializer # 序列化类 class BookDetailView(RetrieveUpdateDestroyAPIView): queryset = Book.objects.all() # 要序列化的数据 serializer_class = BookSerializer # 序列化类
视图集
ModelViewSet= # 5个视图扩展类---》每个类里有一个方法 list,create,retrieve,destory,update mixins.CreateModelMixin, mixins.RetrieveModelMixin, mixins.UpdateModelMixin, mixins.DestroyModelMixin, mixins.ListModelMixin, # ViewSetMixin:特点,只要继承它,路由写法变了,变成了:as_view({'get': 'list', 'post': 'create'}---》请求方式和方法的对应关系 # generics.GenericAPIView 2个视图基类中得一个 GenericViewSet -ViewSetMixin+GenericAPIView # ViewSetMixin 源码分析 class @classonlymethod def as_view(cls, actions=None, **initkwargs): # actions={'get': 'list', 'post': 'create'} def view(request, *args, **kwargs): # method:get action:list for method, action in actions.items(): # handler就是list handler = getattr(self, action) #视图类反射有没有list # 反射:把get变成了list setattr(self, method, handler) return self.dispatch(request, *args, **kwargs) return csrf_exempt(view) # 如果请求来了,会执行 view(request)
# 视图集中的类 -ModelViewSet -ReadOnlyModelViewSet -ViewSetMixin -ViewSet :ViewSetMixin+APIView -GenericViewSet:ViewSetMixin+GenericAPIView # 以后你想继承APIView,但是路由想自动生成,继承ViewSet # 以后想继承GenericViewSet,但是路由想自动生成,GenericViewSet
路由组件
# 1 继承了ViewSetMixin及其子类,路由写法变了 -写法一:path('publish', views.PublishView.as_view({'get': 'wql','post':'login'})), -写法二:自动生成 # 第一步:导入一个路由类 from rest_framework.routers import SimpleRouter, DefaultRouter # 第二步:实例化得到对象 router=SimpleRouter() # router = DefaultRouter() # 它会给每个注册的视图类都生成一个根路由
# 第三步:注册路由---》使用视图类注册---》能自动生成路由 router.register('books', views.Boo kView, 'books') router.register('publish', views.BookView, 'publish')
# 第四步:在总路由中注册 方式二 path('', include(router.urls)),
# 第四步:在总路由中注册 方式一 # urlpatterns+=router.urls # action装饰器的用法 -之前自动生成路由只能生成 {'get': 'list', 'post': 'create'},{'get': 'retrieve', 'put': 'update', 'delete': 'destroy'} -使用action装饰器来做对应关系 ''' methods:请求方式 detail: False:是不是带id的路由 http://127.0.0.1/publish/3 http://127.0.0.1/publish True: 生成的路由publish/pk/wql/ url_path:如果不写,就是函数名生成的路径是:http://127.0.0.1/publish/wql url_name:别名 '''
5.1路由
path('publish', views.PublishView.as_view({'get': 'wql','post':'login'})),
5.2视图类
class PublishView(ViewSetMixin,APIView): # 路由写法变了,变成映射关系了,在视图类中可以写任意的方法名 def lqz(self,request): return Response('wql') def login(self,request): return Response('login')
补充
1 # 浏览器输入地址 www.taobao.com----》dns(本地,网络)(host文件)查找 ip地址--》向这个地址的80端口发送请求(没写端口就是80) 2 序列化的时候,如果传了many=True,生成的序列化类对的对象是ListSerializer的对象 3 assert 断言 本质就是if+抛异常 a = 11 # if a!=10: # raise Exception('a必须是10,不然不能往后走') assert a == 10, ('a必须是10,不然不能往后走') print(a) 4 列表相加
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 震惊!C++程序真的从main开始吗?99%的程序员都答错了
· winform 绘制太阳,地球,月球 运作规律
· 【硬核科普】Trae如何「偷看」你的代码?零基础破解AI编程运行原理
· 上周热点回顾(3.3-3.9)
· 超详细:普通电脑也行Windows部署deepseek R1训练数据并当服务器共享给他人