Django系列 2:request与路由匹配的过程

2020-11-21  本文已影响0人  dingxutao

在上篇文章中我们我们已经已经知道request在Django中的过程,如果没有看过上篇文章 Django系列:请求的生命周期(1)的,可以先看一下,已对你对本文有更好的理解。

本文着重讲解当一个request来到Django后, 是如何进行视图函数或视图类的匹配的,以及如何正确调用请求方法,下图就是当request进来时所要进行请求解析的入口源码:

那么 self.resolve_request(request) 这句代码的背后究竟干了什么了呢?

我们在开发新应用的过程中,都需要用到 urls.py 这个路由文件,那么我们先了解一下:include,path, re_path 三个函数都是干什么的:

注意:from django.conf.urls import url 中url函数将会被弃用,所以我们不讨论该函数

from django.urls import re_path, path, include

(1):def include(arg, namespace=None)

参数:

arg:tuple | module,本质上是一个包含 urlpatterns 变量的 urls.py 文件模块

    (a): 当为元组时, 必须是二元素元素,args=(urlconf_module,                     app_name),urlconf_module是定向应用或根项目 urls.py 文件的路径,app_name可以是应用名称;

    (b): 当为模块变量是,其模块是包含有 urlpatterns 的变量

namespace:命名空间,当namespace存在时,app_name必须存在

返回值:(urlconf_module, app_name, namespace) 三元组;urlconf_module 是一个模块变量

(2):path 和 re_path 都是路由系统,只不过 re_path 的路径包含正则表达式而已

(3):当整个Django项目启动后,会根据ROOT_URLCONF的配置,形成一个链式的路由系统,请求匹配最终需要该类:django.urls.resolvers:URLResolver 解析,经调试候在代码的样子大致如下:

(4):好了, 现在我们知道请求解析匹配的类,那么我们具体看看是如何匹配的,在(1)处时,已经知道请求的入口了,就是这句代码:

callback, callback_args, callback_kwargs =self.resolve_request(request)

resolve_request 方法的可信在这2句代码,请看下图:

resolver = get_resolver():resolver 其实就是(3)提到的URL解析类:django.urls.resolvers:URLResolver, resolver = <URLResolver 'config.urls' (None:None) '^/'>, 对于resolver 的 urlpatterns 属性,它又是链式的。

resolver.resolve(request.path_info),它返回的是一个 django.urls.resolvers:ResolverMatch类,其实ResolverMatch是一个元组,因为它实现了 __getitem__ 魔法函数, 返回值就是 :

callback: 视图类, CBV.as_view() 函数

callback_args:请求的非关键字参数

callback_kwargs :请求的关键字参数 

至此,我们知道请求最终解析后的样子,那么他最终执行视图函数返回Response实例:

response = wrapped_callback(request, *callback_args, **callback_kwargs)

(5):当试图函数执行后,经过 模板中间件、response渲染、异常中间件后,response 再次执行中间件的process_response方法,注意是逆序执行,最终到达第一个中间件后返回wsgi或uwsgi服务器,然后返回给浏览器。

上一篇下一篇

猜你喜欢

热点阅读