1. 定义路由本质

比如在url.py定义以下路由,浏览器中输入http://192.168.0.1:8000/user/2003-04-21可以访问

意味着此url http://192.168.0.1:8000/user/2003-04-21url.py里的路由们做了路由匹配

如果匹配成功找到相应的试图函数

源码解析

ctrl+鼠标左键 点进re_path,会发现

如果re_path = partial(....)

那么上面的url re_path() = partial(), 这里的_path 和Pattern代表什么接着往后看

partial偏函数:

作用:_path()里面规定了要传递一下参数,而path = partial(...)里面传递了Pattern=RegxPatten

那么我们在使用时即不需要再次传递Pattern


点击_path 恍然大悟,_path接收route, view, kwargs, name,Pattern

route view 在url.py已经规定好了, Pattren在partial函数中规定好了

那么这里url.py里的re_path函数就可以写成

partial(_path(route, view, kwargs=None, name=None, Pattern=None), Pattern=RegexPattern)
替换掉route 和 view
partial(_path(r'user/(\d{4})-(\d{2})-(\d{2})', view.user, kwargs=None, name=None, Pattern=None), Pattern=RegexPattern)
因为partial是偏函数,最后路由变成
_path(r'user/(\d{4})-(\d{2})-(\d{2})', view.user, kwargs=None, name=None, Pattern=RegexPattern)

点进_path看看这个函数干啥呢

现在这是我们的路由

re_path(r'user/(\d{4})-(\d{2})-(\d{2})', view.user)就变成了下面这样

URLPattern(Pattern(r'user/(\d{4})-(\d{2})-(\d{2})', name=None, is_endpoint=True), view.user, None, None),

因为规定Pattern = RegxPattern

URLPattern(RegxPattern(r'user/(\d{4})-(\d{2})-(\d{2})', name=None, is_endpoint=True), view.user, None, None),

两个类的关系 URLPattern RegexPattern

URLPattern默认接受四个参数并封装

封装后的

self.pattern = RegxPattern(r'user/(\d{4})-(\d{2})-(\d{2})', name=None, is_endpoint=True)

self.callback = view.user

其他为None

最后到了RegxPattern类,这里会去封装self._regex

上面传递了RegxPattern(r'user/(\d{4})-(\d{2})-(\d{2})', name=None, is_endpoint=True)

那么self._regex= r'user/(\d{4})-(\d{2})-(\d{2})'

两层封装结束,现在

URLPattern.pattern._regex = r'user/(\d{4})-(\d{2})-(\d{2})'

self.callback = view.user

2.路由匹配流程

-启动项目

-请求到来

http://127.0.0.1:8000/user/

与定义好的路由逐一匹配

那么路由是怎么找到的呢

wsgi请求入口

往下找到wsgiHandler函数里的__call__方法,

此函数为类方法入口

environ为每次请求时浏览器携带的参数

request = self.request_class(environ)
因为规定
request_class = WSGIRequest
所以
request = WSGIRequest(environ)

封装好后可以通过request.POST request.GET request.method request.FILES 拿到想要的数据

request.path_info 可以拿到地址

比如 https://192.168.0.1:8000/user/, path_info 可以拿到/user/

下一步通过request 进行函数处理拿到response, 点进get_response

点进get_response发现又进行了_middleware_chain处理

再点 _middleware_chain = handler

往上翻找handler到底是什么

传入的mw_instance

mw_instance = middleware(adapted_handler)

又绕回到上面的adapt_method_mode方法

所以adapted_handler = handler

现在确定handler就是_get_response

下图描述有误,匹配后

callback为完整路径, callback_args为浏览器url传递进的参数

比如url.py规定 re_path('/user/(\d+)')

如访问url为 http://192.168.0.1:8000:/user/123

callback为路经所在内存地址, args为123

现在具体观察这个方法

这里分成两部分查看

1. get_resolver()

进一步查看_get_cached_resolver(urlconf)

URlResolver分析完毕,下面看URLResolver里的resolve方法, 图片注释有误

import_module相当于 from day006 import urls

urls.py被导入之后,patterns = urls.urlpatterns

然后就可进行循环

从最开始的讲解

urls.py里的每一条数据

比如 path('user/(\d{4})-(\d{2})-(\d{2})', view.user)也就等于

URLPattern(RegxPattern(r'user/(\d{4})-(\d{2})-(\d{2})', name=None, is_endpoint=True), view.user, None, None),

每次for pattern做匹配的时候也就是相当于

URLPattern(RegxPattern(r'user/(\d{4})-(\d{2})-(\d{2})', name=None, is_endpoint=True), view.user, None, None),

找到它下面的resolve方法 传入'user/'做验证

这里的resolve就在URLPattern.resolve(new_path)

因为一个re_path() = URLPattern(RegxPattern(r'user/(\d{4})-(\d{2})-(\d{2})', name=None, is_endpoint=True), view.user, None, None),

URLPattern()这里传递的第一个参数为RegxPattern(r'user/(\d{4})-(\d{2})-(\d{2})'

所以self.pattern = RegxPattern(r'user/(\d{4})-(\d{2})-(\d{2})'

这里return的返回值

day-3 路由底层源码的更多相关文章

  1. 7月3日下午 微擎芸众商城 设计思路 - laravel路由底层源码解读

    学习参考文章 https://learnku.com/articles/13622/the-principle-of-laravel-routing-execution <?phpnamespa ...

  2. BAT资深工程师 由浅入深分析 Tp5&Tp6底层源码 - 分享

    BAT资深工程师由浅入深分析Tp5&Tp6底层源码 第1章 课程简介 本章主要让大家知道本套课程的主线, 导学内容,如何学习源码等,看完本章要让小伙伴觉得这个是必须要掌握的,并且对加薪有很大的 ...

  3. BAT资深工程师由浅入深分析Tp5&Tp6底层源码☆

    第1章 课程简介 本章主要让大家知道本套课程的主线, 导学内容,如何学习源码等,看完本章要让小伙伴觉得这个是必须要掌握的,并且对加薪有很大的帮助. 第2章 [TP5灵魂]自动加载Loader 深度分析 ...

  4. Android开发之漫漫长途 Ⅵ——图解Android事件分发机制(深入底层源码)

    该文章是一个系列文章,是本人在Android开发的漫漫长途上的一点感想和记录,我会尽量按照先易后难的顺序进行编写该系列.该系列引用了<Android开发艺术探索>以及<深入理解And ...

  5. 为什么很多类甚者底层源码要implements Serializable ?

    为什么很多类甚者底层源码要implements Serializable ? 在碰到异常类RuntimeException时,发现Throwable实现了 Serializable,还有我们平进的ja ...

  6. List-LinkedList、set集合基础增强底层源码分析

    List-LinkedList 作者 : Stanley 罗昊 [转载请注明出处和署名,谢谢!] 继上一章继续讲解,上章内容: List-ArreyLlist集合基础增强底层源码分析:https:// ...

  7. List-ArrayList集合基础增强底层源码分析

    List集合基础增强底层源码分析 作者:Stanley 罗昊 [转载请注明出处和署名,谢谢!] 集合分为三个系列,分别为:List.set.map List系列 特点:元素有序可重复 有序指的是元素的 ...

  8. 从底层源码浅析Mybatis的SqlSessionFactory初始化过程

    目录 搭建源码环境 POM依赖 测试SQL Mybatis全局配置文件 UserMapper接口 UserMapper配置 User实体 Main方法 快速进入Debug跟踪 源码分析准备 源码分析 ...

  9. Java泛型底层源码解析-ArrayList,LinkedList,HashSet和HashMap

    声明:以下源代码使用的都是基于JDK1.8_112版本 1. ArrayList源码解析 <1. 集合中存放的依然是对象的引用而不是对象本身,且无法放置原生数据类型,我们需要使用原生数据类型的包 ...

  10. 2018.11.20 Struts2中对结果处理方式分析&struts2内置的方式底层源码剖析

    介绍一下struts2内置帮我们封装好的处理结果方式也就是底层源码分析 这是我们的jar包里面找的位置目录 打开往下拉看到result-type节点 name那一列就是我们的type类型取值 上一篇博 ...

随机推荐

  1. LeetCode 双周赛 102,模拟 / BFS / Dijkstra / Floyd

    本文已收录到 AndroidFamily,技术和职场问题,请关注公众号 [彭旭锐] 提问. 大家好,欢迎来到小彭的 LeetCode 周赛解题报告. 昨晚是 LeetCode 双周赛第 102 场,你 ...

  2. Python 使用QQ 邮箱进行发送邮件及经验总结

    今天我带大家实现下简单的发邮件功能.我们要使用到email和smtplib库,这两个库时python自带的,直接import 引用就好了,实现环境python3.6. 对QQ邮箱进行开启SMTP服务 ...

  3. 因果推断-Caual Inference

    两种形式 Reduced Form:Let data speak itself,主要采用regression等方法 Structure Approach:Data only can never rev ...

  4. Tars-Cpp 协程实现分析

    作者:vivo 互联网服务器团队- Ye Feng 本文介绍了协程的概念,并讨论了 Tars Cpp 协程的实现原理和源码分析. 一.前言 Tars 是 Linux 基金会的开源项目(https:// ...

  5. JS逆向实战13——某市公共资源交易中心Cookie混淆加密

    "本文地址:https://www.cnblogs.com/zichliang/p/17346860.html 目标网站 aHR0cDovL2xkZ2d6eS5obmxvdWRpLmdvdi ...

  6. 如何将 Spire.Doc for C++ 集成到 C++ 程序中

    Spire.Doc for C++是一个专业的 Word 库,供开发人员在任何类型的 C++ 应用程序中阅读.创建.编辑.比较和转换 Word 文档. 本文演示了如何以两种不同的方式将 Spire.D ...

  7. Driver8833电机驱动模块的使用(STM32为主控)

    一.硬件 STM32C8T6.STLINK下载器 Driver8833:TI公司的DRV8833是双桥马达驱动器解决方案,包括有两个H桥驱动器,可驱动两个DC电刷马达,或一个步进马达, 螺线管和其它电 ...

  8. vue全家桶进阶之路45:Vue3 Element Plus el_button组件

    在 Vue 3 中,Element Plus 的 ElButton 组件提供了多种按钮类型和属性,可以用于实现不同的交互效果.下面是 ElButton 常用的作用和属性: 作用: 用于在页面上添加交互 ...

  9. Vue 全局避免按钮重复点击

    这里用到的 Vue.directive 自定义指令 自定义指令是对普通DOM元素进行的底层操作,它是一种有效的的补充和扩展,不仅可以用于定义任何的dom操作,并且是可以复用的 在 main.js 中写 ...

  10. drf——权限、认证源码分析、过滤、排序、分页

    权限.认证源码(了解) 权限源码 # 继承了APIView才有的--->执行流程--->dispatch中的三大认证 self.initial(request, *args, **kwar ...