Django 基于类的视图(CBV)执行流程 CBV 源码分析

时间:2023-12-19 09:24:50

一、CBV(基于类的视图)

视图是可以调用的,它接受请求并返回响应,这不仅仅是一个函数,Django提供了一些可以用作视图的类的例子,这些允许您通过继承或mixin来构建视图并重用代码。

基本示例

Django提供了基本的视图类,它将适用于广泛的应用程序。所有的视图类都继承自View该类,它处理将视图链接到URL,HTTP方法调用和其他简单的功能。

在URLconf中简单使用

BookView是一个视图类,而不是一个函数,所以我们将URL指向as_view()类方法,它为基于类的视图提供了类似于函数的条目:

from django.conf.urls import url
from django.contrib import admin
from bookmanage import views urlpatterns = [
url(r'^admin/', admin.site.urls),
url(r'books/',views.BookView.as_view()),
]

BookView类

继承现有的View并覆盖父类中的方法。

from django.shortcuts import render,HttpResponse
from django.views import View class BookView(View): def get(self,request):
return HttpResponse('get....') def post(self,request):
return HttpResponse('post....')

二、View

主要的基于类的基本视图,所有其他基于类的视图都是继承自这个View类,导入方式

from django.views import View

浏览器向服务器发送一个get请求,基于类的视图执行流程

首先,django启动,会执行视图类下的as_view():

urlpatterns = [
url(r'^admin/', admin.site.urls), # as_view() 执行类的as_view()
url(r'books/',views.BookView.as_view()),
]

as_view():由于BookView没有实现as_view()方法,会调用父类(View)中的as_view()方法:

@classonlymethod
def as_view(cls, **initkwargs):
"""
请求响应过程的主要入口点
"""
......
def view(request, *args, **kwargs):
self = cls(**initkwargs)
if hasattr(self, 'get') and not hasattr(self, 'head'):
self.head = self.get
self.request = request
self.args = args
self.kwargs = kwargs #view方法返回self.dispatch()方法的返回值
return self.dispatch(request, *args, **kwargs)
view.view_class = cls
view.view_initkwargs = initkwargs
update_wrapper(view, cls, updated=())
update_wrapper(view, cls.dispatch, assigned=()) #返回view方法内存地址,当用户访问books/的时候,会调用改view()方法
return view

self.dispatch():由于BookView没有实现dispatch(),所以会调用View类的dispatch()

主要作用:view视图的一部分 - 接受request 参数和参数的方法,并返回HTTP响应。

检测HTTP请求方法,并尝试委托给匹配HTTP方法的方法,一个GET将被委托给get(),一POSTpost(),等等。

def dispatch(self, request, *args, **kwargs):
# 尝试把用户请求分发到正确的方法,如果方法不存在,遵从错误的处理程序
# 如果请求方法不在已批准的列表中,也会遵从错误的处理程序
if request.method.lower() in self.http_method_names:
#使用反射,例如用户访问是get请求
# handler = getattr(BookView,"get",错误处理程序)
handler = getattr(self, request.method.lower(), self.http_method_not_allowed)
else:
handler = self.http_method_not_allowed # return get(request,*args, **kwargs)
return handler(request, *args, **kwargs)

URLconf中的变化

此时URLconf的配置
url(r'books/',views.BookView.as_view()),
等价于
url(r'books/',view) # 由于view return self.dispatch()
等价于
url(r'books/',self.dispatch())
等价于
url(r'books/',handler(request,*args,**kwargs)
--> url(r'books/',get(request,*args,**kwargs)
--> url(r'books/',post(request,*args,**kwargs)
等等