python|web应用框架|使用类装饰器注册路由

本文涉及的产品
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
简介: python|web应用框架|使用类装饰器注册路由

之前我们介绍了如何利用WSGI写一个简单的web应用框架。写出来之后,项目虽然能用,如果还没有看过上述文章的,建议先看下,以便做到承上启下:

python|写一个简单的web应用框架: juejin.cn/post/722635…


本文的python环境为:

image.png

在上述文章中,我们调用框架后,在定义路由的时候,使用的代码如下:

myWeb.Routes("/index",index)
myWeb.Routes("/123",d345)


如上代码,虽然能够完成工作,但是不够美观。我们想要我们的框架像flask那样,使用装饰器来定义路由,例如,这样:

routes(path="/123",methods="post")
def helloWold(r):
    return (200,"hello world")

这样的话,就不用再额外使用Routes将路由和函数绑定了。



类装饰器应该怎么写?

要实现上述功能,需要用到类装饰器,这里需要先简单阐述一下如何使用类装饰器,我们之前使用的都是函数来做的装饰器。


再探类私有方法

类装饰器需要重写__init__方法和__call__方法。如之前文章所述,前者在实例化类为对象的时候进行调用,而后则则是在调用对象的时候才会调用。


这里写一个简单的案例,来复习一下__init____call__方法。


image.png


上述代码,就写了一个类testClass,在该类中,我们定义了__init__方法和__call__方法,我们为每隔一方法打印了一条输出语句。


最后我们定义了一个类testClass实例test1。而后再运行该实例。

运行后效果如下:

image.png

通过输出,我们可以发现,在将类实例化为对象的时候,就启动了__init__方法,而__call__方法,则需要调用实例的时候,才能运行。



装饰器调用方法


我们目前看到的装饰器调用方法,都是在函数前面定义@而后跟上装饰器名称,如果有参数的话,会跟上参数,但是这都是语法糖的效果。


看一个装饰器例子:

image.png


上述代码,我们定义了一个装饰器decorator,该装饰器仅是在调用函数前后打印输出一些内容。而后再test3函数上面,使用@来调用装饰器。最后执行函数test3

代码执行效果如下:

image.png

上述代码中,所谓的语法糖就是@后面的语句,下面我们将不是用语法糖,来实现一下这个操作。

image.png

该代码,我们不使用python语法糖,所以我们定义手动将函数test3作为参数,传入decorator装饰器中,装饰器返回wrapper函数,我们将使用f来接收,最后我们通过调用f函数来实现装饰器的调用。

代码执行结果如下:

image.png

还有一种情况是,如果装饰器带参数了,这个时候需要在装饰器外部在定义一层函数,用于接收参数,例如代码如下:

image.png

上述代码,使用了re来接收装饰器本身的参数,其他的和上述一致,我们执行代码,结果如下:

image.png

这里之所以强调装饰器,是因为我们想写函数式的装饰器来接收参数。



类作为装饰器

类善用私有方法,也可以实现装饰器的效果,例如:

image.png

上述代码,我们使用类的私有方法,来定义装饰器,上述代码如果不是语法糖的话,那么下面这段代码,应该如何调用呢?

@decorator
def test3():
    print("哈哈哈")


如果不是语法糖的话,是不是应该是

f  = decorator(test3)
f()


所以,需要使用__init__方法来接收函数,而在call方法中,我们直接写代码即可,中间需要调用从__init__传过去的方法。

代码执行后,结果如下:

image.png

如果装饰器需要传参,传参这个动作就是类+(),就会执行底层方法__call__,所以,参数需要在__init__中接收,而函数需要在__call__中去接收,代码如下:

image.png


这个代码写的相对复杂,这里解析一下,我们将上述装饰器拆分为普通函数,代码如下:

f = decorator(x=3,y=5)
f1 = f(test3)
print(f1(3,5))

好了,现在看起来是不是感觉好多了呢?我们调用decorator(x=3,y=5),该代码是将类初始化为对象,所以需要在__init__方法中去解析传过来的xy,而f(test3),则是对象的调用了,需要需要去__call__找到传入的函数。这个时候__call__里面又是一个闭包函数,将闭包函数返回后,得到f1,最后我们执行f1函数,就先当于执行了wraper()函数,如果需要传参的话,我们直接写入参数即可,因为在闭包函数wrapper中,我们也传参了的: *args,**kwargs


所以上述代码,使用@语法糖装饰器的话,执行结果如下:

image.png


如何注册路由


有了上面类装饰器的铺垫,再来写解析路由的话,就非常容易了,我们首先要确定,装饰器中应该传入什么样的值?作为绑定路由信息来说,我们最需要知道的是路由值和请求方法,所以我们想定义绑定函数类似于如下:

routes(path="/123",methods="post")
def helloWold(r):
    return (200,"hello world")

那作为routes装饰器,我们需要在其__init__方法中接收pathmethods的值,在__call__方法中获取函数值。


当注册路由之后,我们直接将其存储到字典中,整个注册路由就写完了。


我们可以查看代码:

image.png


如上代码,我们先定义了2个字典:mapGetRoutemapPostRoute来存储GET请求和POST请求的信息。


当进行函数注册的时候,需要获取装饰器传上来的pathmethods,而后根据其请求方法,将数据存储到对应的字典中,若是指定的ALL,则2个都存储。


最后只需要在启动函数application中,对客户端请求进行解析,如果存在注册路由的函数,就直接获取存储的函数值调用就完事了。

image.png


上述代码,我们会对每一个请求进行判断,而后取出相应字典中的函数,并且执行后,将结果返回回去。注意,这里还是有问题的,若请求没有在路由字典中,则func的值为None,就会抛错,我们可以定义一个万金油路由,若匹配不到,就重定向到该路由函数中去,向客户端返回404即可。


改善后的框架代码,我也放到了gitee上: gitee.com/pdudo/golea…

启动wsgi应用,我们直接使用的wsgiref,代码案例如下:

gitee.com/pdudo/golea…



总结


这篇文章,我们介绍了类作为装饰器应该如何调用,最后我们通过该特性,复写了一下我们框架的路由解析,让其路由函数和其他普通函数分开好看一点 。类写装饰器,其实是善用了其私有函数__init____call__。这里要注意一下,装饰器带参数和不带参数对应的类装饰器,都是不同的。





相关文章
|
1天前
|
前端开发 JavaScript Python
Python Web应用中的WebSocket实战:前后端分离时代的实时数据交换
在前后端分离的Web应用开发模式中,如何实现前后端之间的实时数据交换成为了一个重要议题。传统的轮询或长轮询方式在实时性、资源消耗和服务器压力方面存在明显不足,而WebSocket技术的出现则为这一问题提供了优雅的解决方案。本文将通过实战案例,详细介绍如何在Python Web应用中运用WebSocket技术,实现前后端之间的实时数据交换。
8 0
|
1天前
|
XML 前端开发 API
惊艳全场的秘诀!AJAX、Fetch API与Python后端,打造令人惊叹的Web应用!
惊艳全场的秘诀!AJAX、Fetch API与Python后端,打造令人惊叹的Web应用!
5 0
|
1天前
|
运维 负载均衡 安全
深度解析:Python Web前后端分离架构中WebSocket的选型与实现策略
深度解析:Python Web前后端分离架构中WebSocket的选型与实现策略
8 0
|
1天前
|
JSON API 开发者
深入解析Python网络编程与Web开发:urllib、requests和http模块的功能、用法及在构建现代网络应用中的关键作用
深入解析Python网络编程与Web开发:urllib、requests和http模块的功能、用法及在构建现代网络应用中的关键作用
6 0
|
1天前
|
iOS开发 MacOS Python
Python 编程案例:谁没交论文?输出并生成电子表格
Python 编程案例:谁没交论文?输出并生成电子表格
17 9
|
1天前
|
IDE 开发工具 iOS开发
Python编程案例:查找指定文件大小的文件并输出路径
Python编程案例:查找指定文件大小的文件并输出路径
10 3
|
1天前
|
文件存储 iOS开发 MacOS
Python编程案例:文件查找并归类
Python编程案例:文件查找并归类
|
1天前
|
Python
Python编程案例:同一工作簿不同表单特定数据添加到工作簿的另一表单里
Python编程案例:同一工作簿不同表单特定数据添加到工作簿的另一表单里
|
1天前
|
iOS开发 MacOS Python
Python编程案例:根据姓名归档论文、报告
Python编程案例:根据姓名归档论文、报告
|
1天前
|
数据处理 Python
Python编程-利用datetime模块生成当前年份之前指定的间隔所有年份的日期列表和csv文件
Python编程-利用datetime模块生成当前年份之前指定的间隔所有年份的日期列表和csv文件