(五)Flask之深入剖析路由源码

news2025/2/2 15:55:44
  • 路由(Route)这个概念在所有web框架中都非常重要,它是用于定义URL和对应的处理函数(视图)之间的映射关系。通过定义路由,可以使web框架应用程序能够响应不同的URL请求,并执行相应的逻辑。

源码剖析:

在Flask项目(比如下面这个hello world项目)刚启动时,Flask内部肯定会将所有的路由和视图创建对应关系。

from flask import Flask

app = Flask(__name__)


@app.route('/index', methods=['GET', 'POST'])
def index():
    return 'hello world'


if __name__ == '__main__':
    app.run('localhost', 4000)

分析源码:

当程序运行到@app.route('/index', methods=['GET', 'POST'])时,由于@语法糖的存在,会立刻执行route函数,所以进入route函数源码跟踪下:

[外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-akxZeP0V-1689295504389)(https://note.youdao.com/yws/res/18797/WEBRESOURCE72907d442753eb892b3902f227113780)]

app对象的route函数执行会返回decorator函数,即@decorator 等价于 @app.route(‘/index’, methods=[‘GET’, ‘POST’])

@语法糖的存在,会立刻执行@decorator

即decorator(index函数)

【因为程序下方没有执行index()函数,所以不用继续看@语法糖的第二步(在《(三)Flask前置知识栈——装饰器》这篇文章有详解)】

继续看route函数源码,decorator(index函数)对应—>decorator中的参数f就是index函数。

在这里插入图片描述

self.add_url_rule(rule, endpoint, f, **options)

源码中的这一句是关键,功能就是添加路由和视图函数的对应规则。

继续追踪源码,看下这个函数:

【注意add_url_rule()函数在Flask(__name__)这个对象中】

在这里插入图片描述

重点就是上图中红框部分。


简单剖析一波上图的add_url_rule()源码:

add_url_rule()方法是在Flask应用程序中注册URL规则的方法。

def add_url_rule(
    self,
    rule: str,
    endpoint: t.Optional[str] = None,
    view_func: t.Optional[ft.RouteCallable] = None,
    provide_automatic_options: t.Optional[bool] = None,
    **options: t.Any,
) -> None:
	"""
	*   rule:URL规则的字符串表示,对应'/index'。
	*   endpoint:可选参数,指定URL规则的名称(即终点)。
	*   view_func:可选参数,指定与URL规则关联的处理函数(视图函数)。
	*   provide_automatic_options:可选参数,控制是否自动提供OPTIONS方法。
	"""

首先,方法会对参数进行处理和验证:

# 如果没有指定`endpoint`,则根据`view_func`自动生成一个。
if endpoint is None:
    endpoint = _endpoint_from_view_func(view_func)
options["endpoint"] = endpoint


# 从options参数中弹出`methods`字段,并根据不同情况设置默认值为"GET"。最后,将`methods`转换为大写,并确保它是一个集合。
methods = options.pop("methods", None)
if methods is None:
    methods = getattr(view_func, "methods", None) or ("GET",)
if isinstance(methods, str):
    raise TypeError("Allowed methods must be a list of strings, for example: @app.route(..., methods=['POST'])")
methods = {item.upper() for item in methods}

接下来,根据view_func的属性和参数值,设置是否自动提供OPTIONS方法(文末会简单讲下这个是啥玩意):

# 如果没有指定`provide_automatic_options`,则从`view_func`的属性中获取。
if provide_automatic_options is None:
    provide_automatic_options = getattr(view_func, "provide_automatic_options", None)

# 如果仍然未指定,则根据methods是否包含"OPTIONS"来决定是否自动提供OPTIONS方法。
if provide_automatic_options is None:
    if "OPTIONS" not in methods:
        provide_automatic_options = True
        required_methods.add("OPTIONS")
    else:
        provide_automatic_options = False

接下来,将必需的方法添加到methods集合中,并创建一个Rule对象来表示URL规则:

# 将必需的方法合并到`methods`集合中(文末会讲这样的作用)
methods |= required_methods 

# 使用`url_rule_class`创建一个`Rule`对象来表示URL规则
rule = self.url_rule_class(rule, methods=methods, **options)
rule.provide_automatic_options = provide_automatic_options

# 将rule添加到应用程序的URL映射中。
self.url_map.add(rule)

最后,如果指定了view_func,则将其关联到endpoint,并进行冲突检查:

if view_func is not None:
    old_func = self.view_functions.get(endpoint)
    if old_func is not None and old_func != view_func:
        raise AssertionError(f"View function mapping is overwriting an existing endpoint function: {endpoint}")
    self.view_functions[endpoint] = view_func

如果之前已经有一个与endpoint关联的处理函数,则会引发AssertionError(熟悉吗?在装饰器那篇引出的那个问题Flask就是在这抛出的异常!!!)。

总结:add_url_rule()方法用于向Flask应用程序添加URL规则。它解析和验证参数,设置默认值,并创建和注册Rule对象来表示URL规则。然后,将处理函数与URL规则关联,并进行必要的冲突检查。


继续按①追踪源码:

在这里插入图片描述
在这里插入图片描述

鄙人简单翻译一下:

一个Rule代表一个URL模式。“Rule”有一些选项会改变它的行为方式,并传递给“Rule”构造函数。

注意,除了规则字符串之外,所有参数必须为关键字参数,以免在Werkzeug升级时破坏应用程序。

说大白话就是:

Rule类对URL模式进行解析和存储。Rule类表示一个URL规则,并包含与之相关联的信息,如URL模式、HTTP方法、处理函数等。

继续按②追踪源码:

进入url_map:

在这里插入图片描述

进入url_map_class

在这里插入图片描述

鄙人简单翻译一下:

map类存储所有URL规则和一些配置参数。一些配置值仅存储在’ Map '实例,因为它们影响所有规则,其他只是默认值

并且可以为每个规则重写。注意,必须指定所有除了作为关键字参数的’ rules '之外的参数!

总之,一句话:

Map是一个URL映射器,它用于存储和管理应用程序中所有的URL规则。

继续追踪Map类中的add()函数(注释讲的很清楚了~):

在这里插入图片描述

继续按③追踪源码(文末会简单讲一下这行源码):

在这里插入图片描述

上面是三条线分开剖析的源码,只是简单分析,后续会慢慢深入。

view_functionsMapRule放一起看看:

view_functionsMapRule之间存在密切的关系。它们是Flask框架中用于处理URL路由的重要组件。

  • view_functions是Flask应用程序对象的一个属性,存储了应用程序中所有路由终点(endpoint)与对应视图函数(view function)之间的映射关系。它是一个字典,键是路由的终点名称,值是对应的视图函数。通过app.view_functions可以访问和操作这个字典。

  • Map是一个URL映射器,用于存储和管理应用程序中的所有URL规则。它是werkzeug.routing.Map类的实例,通过app.url_map可以访问应用程序的Map对象。Map类提供了添加、查找和操作URL规则的方法,包括add()match()iter_rules()等。

  • Rule是URL规则的表示,它定义了一个URL模式、绑定的视图函数、可接受的请求方法等信息。Rule类是werkzeug.routing.Rule类的实例,作为Map对象的一部分使用。每个Rule对象都有一个唯一的终点(endpoint),用于标识该规则。Map类使用Rule对象来存储和管理URL规则。

关系如下:

  • 在定义Flask应用程序时,我们使用@app.route()装饰器来创建URL规则,并将路由终点和视图函数关联起来。这些URL规则会被存储在Map对象中,通过app.url_map访问。
  • 在调用app.route()装饰器时,Flask框架内部会创建一个Rule对象,表示这个URL规则。然后,Rule对象会添加到Map对象中,构成了整个URL映射关系。
  • Map对象使用view_functions属性来查找和调用对应的视图函数。当请求到达时,Map会根据请求的URL路径和HTTP方法,在内部查找匹配的Rule对象,并从中获取绑定的视图函数。
  • 通过view_functions字典和Map对象的结合,Flask能够实现自动路由和请求处理的功能。它将URL的终点(endpoint)作为键,从view_functions中查找对应的视图函数;同时,根据请求的URL和方法,从Map中查找匹配的Rule对象,并提取其中的视图函数进行调用。

可见,view_functionsMapRule是Flask框架中处理URL路由的关键组件,它们相互协作,才能实现灵活的路由管理和请求处理机制。

综上—添加路由的两种方式:

from flask import Flask

app = Flask(__name__)
app.config.from_object()


# 方式一    (就用这个!)
@app.route('/index', methods=['GET', 'POST'])
def index():
    return 'index'


def order():
    return 'order'


# 方式二
app.add_url_rule('/order', view_func=order)

if __name__ == '__main__':
    app.run('localhost', 4000)

拓展:

(1)深入provide_automatic_options参数

provide_automatic_options参数用于控制是否自动提供OPTIONS方法。

在HTTP协议中,OPTIONS方法用于获取指定URL支持的请求方法。Flask框架默认情况下会自动处理OPTIONS请求,并返回一个包含允许的请求方法的响应。

provide_automatic_options参数为True时,表示应用程序会自动提供OPTIONS方法的支持。如果请求的URL匹配到了一个路由规则,但是没有显式定义OPTIONS方法的处理函数,Flask会自动生成一个OPTIONS方法的处理函数来响应该请求。

provide_automatic_options参数为False时,表示应用程序不会自动提供OPTIONS方法的支持。即使请求的URL匹配到了一个路由规则,但是没有显式定义OPTIONS方法的处理函数,Flask也不会自动生成OPTIONS方法的处理函数。这意味着如果客户端发送了一个OPTIONS请求,而且没有相应的处理函数来处理该请求,Flask将返回一个405 Method Not Allowed的响应。

通过设置provide_automatic_options参数,开发者可以灵活地控制应用程序是否自动提供OPTIONS方法的支持,以满足具体需求。

(2)methods |= required_methods的作用

methods |= required_methods是在Flask中的路由定义过程中用于合并必需的方法(required methods)到可接受的方法(methods)集合的操作。

在Flask中,可以通过将请求方法作为装饰器参数传递给路由函数来指定允许的请求方法。例如:

@app.route('/', methods=['GET', 'POST'])
def index():
    # 处理GET和POST请求
    pass

在上述示例中,路由函数index()被定义为只接受GET和POST请求。但有时,可能还需要处理其他方法,例如OPTIONS或PUT等。这就是required_methods的作用。

required_methods是一个集合,包含了一些应该始终添加到可接受的方法集合中的方法。这些方法通常是特定于应用程序的、必须支持的方法。

通过使用按位或(OR)运算符|=,将required_methods集合合并到methods集合中,并将结果赋值给methods。这样就确保了methods集合中包含了所有必需的方法。

例如,如果methods集合已经包含了[‘GET’],而required_methods集合包含了[‘OPTIONS’],那么执行methods |= required_methods后,methods集合将变为[‘GET’, ‘OPTIONS’]。

这样做的目的是确保路由能够处理所有必需的请求方法,从而提供更全面的HTTP方法支持。

(3)深入self.view_functions

self.view_functions是一个字典,用于存储Flask应用程序中所有路由终点(endpoint)与对应视图函数(view function)之间的映射关系。

它是在Flask应用程序对象的初始化过程中创建的,并作为一个实例属性存在。self.view_functions字典的定义如下:

self.view_functions: Dict[str, Callable] = {}

这意味着self.view_functions是一个键值对形式的字典,其中键是路由的终点名称,值是对应的视图函数。具体来说,键是字符串类型,值是可调用对象(callable),通常是Python函数。

当使用@app.route()装饰器定义路由时,Flask框架会自动将路由终点和对应的视图函数添加到self.view_functions字典中。这样,在接收到请求时,Flask可以根据路由终点从字典中找到对应的视图函数,并执行相应的逻辑。

实战——如何使用self.view_functions字典:

from flask import Flask

app = Flask(__name__)


@app.route('/')
def demo():
    return 'Hello, World!'


@app.route('/index')
def index():
    return 'index'


if __name__ == '__main__':
    print(app.view_functions)

运行就会输出app.view_functions字典的内容:

{
	'static': <function Flask.__init__.<locals>.<lambda> at 0x000001A3334258B0>, 
	'demo': <function demo at 0x000001A3334364C0>, 
	'index': <function index at 0x000001A333436550>
}

可以通过访问app.view_functions来查看和操作Flask应用程序中注册的视图函数,例如获取所有终点名称、获取对应的视图函数等。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/762442.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

老码农的管理拙见

【引子】 尽管自己从业了20多年&#xff0c;也曾管理过从几个人到几百人的团队&#xff0c;但个人非常不愿意或者不敢讨论团队管理的问题&#xff0c;因为管理是以结果为导向的&#xff0c; 具有后验的特征&#xff0c;而且管理中面对的最大复杂性是人&#xff0c;每个人都是不…

安装pycocotools报错“fatal error: Python.h: No such file or directory“ | 已解决

记录一段报错&#xff1a; pycocotools/_mask.c:6:10: fatal error: Python.h: No such file or directory#include "Python.h"^~~~~~~~~~compilation terminated./tmp/pip-build-env-crtws8v6/overlay/lib/python3.8/site-packages/Cython/Compiler/Main.py:369: F…

mssql 以xml类型为存储过程传递不确定数量的参数

mssql 以xml类型传递不确定数量的参数 存储过程xml 处理在存储过程中参数在存储过程中使用 xml 作为参数存储过程 相信各位小伙伴在使用数据库的过程中,或多或少的建立了一些存储过程,并且带有一些参数,用来增加存储过程的适用性。 类似老顾的截图这样的,通常,我们需要将…

华为云CodeArts Check IDE插件体验之旅

1 开发者的思考 近年来&#xff0c;ChatGPT的来临像一场突然出现的风暴&#xff0c;程序员是否马上被取代的担忧出现在媒体上了&#xff0c;作为软件开发小白&#xff0c;前不久我也陷入了这样的深思之中&#xff0c;但认真的想了下&#xff0c;ChatGPT就如自动驾驶一样&#…

MySQL的安装与配置

今天要和大家唠唠关于数据库的那些事儿&#xff01;按照加哥一贯的调性&#xff0c;咱还是从花边八卦聊起。先来简 单地梳理一下数据库、MySQL发展的时间线&#xff1a; 1970年&#xff0c;在IBM公司工作的数学家 E.F.Codd 发表了数学论文《大型共享数据库的关系数据模型》 &am…

每日站会如此简单,为什么总是开不好?

美式足球或橄榄球等运动的球队&#xff0c;会在每场比赛上场前聚在一起开个短会。这种临场短会能让整个球队的成员在比赛过程中互通信息、相互协作。 每日站会是敏捷开发的重要流程之一。对于团队而言&#xff0c;每日站会与这种赛前短会类似&#xff0c;让每个成员都了解到团…

2023年上海/成都/深圳CSPM-3中级国标项目管理认证招生

CSPM-3中级项目管理专业人员认证&#xff0c;是中国标准化协会&#xff08;全国项目管理标准化技术委员会秘书处&#xff09;&#xff0c;面向社会开展项目管理专业人员能力的等级证书。旨在构建多层次从业人员培养培训体系&#xff0c;建立健全人才职业能力评价和激励机制的要…

Matplotlib坐标轴范围

Matplotlib 可以根据自变量与因变量的取值范围&#xff0c;自动设置 x 轴与 y 轴的数值大小。当然&#xff0c;您也可以用自定义的方式&#xff0c;通过 set_xlim() 和 set_ylim() 对 x、y 轴的数值范围进行设置。 当对 3D 图像进行设置的时&#xff0c;会增加一个 z 轴&#x…

日志收集工具

日志管理的第一步是收集日志数据。日志收集可能是一项具有挑战性的任务&#xff0c;因为某些系统&#xff08;如防火墙、入侵检测系统和入侵防御系统&#xff09;具有生成大量日志数据的 EPS&#xff08;每秒事件数&#xff09;。为了实时收集和处理日志数据&#xff0c;无论日…

【历史上的今天】6 月 7 日:图灵逝世;Kubernetes 开源版本发布;分组交换网络发明者出生

整理 | 王启隆 透过「历史上的今天」&#xff0c;从过去看未来&#xff0c;从现在亦可以改变未来。 今天是 2023 年 6 月 7 日&#xff0c;在 1742 年的今天&#xff0c;普鲁士数学家克里斯蒂安哥德巴赫在写给瑞士数学家莱昂哈德欧拉的通信中&#xff0c;提出了以下的猜想&…

手机APP三维建模不清晰?单反相机接入移动端来袭!

为什么我们需要单反相机接入 当前&#xff0c;手机本身的像素还满足不了考古级的精度要求&#xff0c;考古现场都是用专业相机对文物进行拍照&#xff0c;再把照片导入到电脑进行建模。“考古工作比较细致&#xff0c;有时候一来一回挺费事的。”云端地球的忠实用户&#xff0…

Linux5.16 Ceph集群

文章目录 计算机系统5G云计算第四章 LINUX Ceph集群一、Ceph1.存储基础1&#xff09;单机存储设备2&#xff09;单机存储的问题3&#xff09;商业存储解决方案4&#xff09;分布式存储&#xff08;软件定义的存储 SDS&#xff09;5&#xff09;分布式存储的类型 2.Ceph 简介3.C…

微信公众号登录

整个流程&#xff0c;1.前端调用授权url 接口(创建一个重定向的请求方法&#xff0c;访问自动回调方法wechat.mp.callbackUrl的地址)。2.微信自动回调方法里判断该用户是需要注册还是直接登录(如果直接登录就返回token&#xff09; 是注册还是登录返回到配置文件中的 wechat.mp…

LIMS系统应用于第三方实验室的意义

金现代LIMS实验室管理系统是将实验室业务从线下转到线上&#xff0c;按照 CNAS 规范进行合规化管理&#xff0c;并提升实验室工作效率的信息化系统。随着实验室业务的不断发展&#xff0c;实验室的规模逐渐扩大&#xff0c;样品数量和检测项目不断增加&#xff0c;数据管理需求…

客户报备小程序系统开发制作方案

客户报备系统小程序是一种用于公司业务人员进行客户管理和报备的小程序系统。 客户报备小程序系统主要功能包括以下几个方面&#xff1a; 1、用户注册与登录 提供用户注册和登录功能&#xff0c;确保用户可以使用个人账户登录系统。 2、报备信息录入 对于需要报备的客户&#x…

手把手教你搭建SpringCloud项目(三)集成Eureka服务注册中心

我们首先了解 Eureka&#xff1a;官方介绍 一、了解Eureka 1、什么是服务治理&#xff1f; Springcloud 封装了Netfix公司开发的Eureka模块来实现服务治理。在传统的RPC远程调用中&#xff0c;管理每个服务于服务之间依赖关系复杂&#xff0c;管理复杂&#xff0c;所以需要使…

113. 路径总和 II

113. 路径总和 II 原题链接&#xff1a;完成情况&#xff1a;解题思路&#xff1a;参考代码&#xff1a; 原题链接&#xff1a; 113. 路径总和 II https://leetcode.cn/problems/path-sum-ii/ 完成情况&#xff1a; 解题思路&#xff1a; /**解题思路&#xff1a;* 和之前一…

msys2安装与配置: 在windows上使用linux工具链g++和包管理工具pacman C++开发

文章目录 为什么用这个msys2下载、doc安装&#xff0c;很简单初次运行&#xff0c;做些配置更新软件安装与卸载方法安装必要的软件包设置win环境变量在windows terminal中使用在vscode中使用 为什么用这个msys2 方便windows上的C开发demo&#xff0c;不需要VS了方便C开发安装o…

Redis缓存更新策略以及常见缓存问题

文章目录 一、什么是缓存&#xff1f;二、添加Redis缓存三、缓存更新策略四、缓存穿透五、缓存雪崩六、缓存击穿七、缓存工具封装 一、什么是缓存&#xff1f; 缓存就是数据交换的缓冲区(Cache)&#xff0c;是存储数据的临时地方&#xff0c;一般读写性能较好&#xff0c;常见…

Mac环境配置(Java)----使用bash_profile进行配置

1、打开软件--终端 2、首先查看本机Java的安装地址&#xff08;系统默认的&#xff09; /usr/libexec/java_home -V 查看到Java8安装的路径如下&#xff1a; 3、如果是第一次配置环境变量&#xff0c;使用命令&#xff1a;【touch .bash_profile】创建一个.bash_profile隐藏…