在DRF中,所有drf的操作都是在路由匹配完成后,即视图函数执行前和执行后做文章的。
一、代码准备
演示的视图:
class TestAPIView(APIView):def get(self,request)return Respponse({'code':200,'msg':'测试通过'})
演示的路由:
path('test/',views.TestAPIView.as_view())
演示的请求方式:GET请求
二、视图函数执行的流程
1、调用TestAPIView的as_view方法。
2、TestAPIView不存在as_view方法,调用父类APIView的as_view方法 (简化的源码)
@classmethod
def as_view(cls, **initkwargs):#1、调用APIView父类的as_view方法view = super().as_view(**initkwargs)view.cls = clsview.initkwargs = initkwargs#2、去除掉csrf校验return csrf_exempt(view)
3、Django的View中as_view的逻辑
@classonlymethod
def as_view(cls, **initkwargs):def view(request, *args, **kwargs):self = cls(**initkwargs)self.setup(request, *args, **kwargs)if not hasattr(self, 'request'):raise AttributeError("%s instance has no 'request' attribute. Did you override ""setup() and forget to call super()?" % cls.__name__)return self.dispatch(request, *args, **kwargs)view.view_class = clsview.view_initkwargs = initkwargsreturn view#返回的是闭包view,
#view的功能就是self.dispatch()方法
4、主要是self.dispatch()方法,self是TestAPIView的对象 , 没有dispatch方法,调用父类APIView的dispatch方法 (简化的源码)
def dispatch(self, request, *args, **kwargs):try:#self是TestAPIView的对象,去该对象中去获取请求方式如GET的小写的方法gethandler = getattr(self, request.method.lower(),self.http_method_not_allowed)response = handler(request, *args, **kwargs)except Exception as exc:response = self.handle_exception(exc)self.response = self.finalize_response(request, response, *args, **kwargs)return self.response
5、通过反射,去TestAPIView中获取get方法,执行get方法后,返回执行的结果。
大致的流程:
总结:DRF的视图执行过程,有使用到原生Django的代码。
注意:在源码解析的过程中,要时刻知道self指代是谁的实例对象。