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

本文涉及的产品
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 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__。这里要注意一下,装饰器带参数和不带参数对应的类装饰器,都是不同的。





相关文章
|
8天前
|
小程序 前端开发 中间件
ThinkPHP 配置跨域请求,使用TP的内置跨域类配置,小程序和web网页跨域请求的区别及格式说明
本文介绍了如何在ThinkPHP框架中配置跨域请求,使用了TP内置的跨域类`\think\middleware\AllowCrossDomain::class`。文章还讨论了小程序和web网页在跨域请求格式上的区别,并提供了解决方案,包括修改跨域中间件源码以支持`Origin`和`token`。此外,还介绍了微信小程序跨域请求的示例和web网页前端发送Axios跨域请求的请求拦截器配置。
ThinkPHP 配置跨域请求,使用TP的内置跨域类配置,小程序和web网页跨域请求的区别及格式说明
|
2天前
|
数据安全/隐私保护 Python
探索Python中的装饰器:简化代码,提升效率
【9月更文挑战第32天】在Python编程世界中,装饰器是一个强大的工具,它允许我们在不改变函数源代码的情况下增加函数的功能。本文将通过直观的例子和代码片段,引导你理解装饰器的概念、使用方法及其背后的魔法,旨在帮助你写出更加优雅且高效的代码。
|
1天前
|
开发者 Python
深入理解Python中的装饰器
【9月更文挑战第33天】本文将通过浅显易懂的语言和生动的比喻,带领读者走进Python装饰器的奇妙世界。我们将从装饰器的基本概念出发,逐步探索其背后的原理,并通过实际代码示例,展示如何运用装饰器优化我们的代码结构。无论你是编程新手还是有一定基础的开发者,这篇文章都将为你打开一扇通往高效编码的大门。
|
1天前
|
存储 缓存 开发者
探索Python中的装饰器:从基础到高级应用
【9月更文挑战第33天】本文将带你走进Python的装饰器世界,从理解其核心概念出发,逐步深入到实现机制与应用场景。我们将通过实际代码示例,展示如何利用装饰器简化代码、增强函数功能,并讨论装饰器的高级用法,如带参数装饰器和装饰器嵌套。无论你是初学者还是有经验的开发者,这篇文章都将为你提供有价值的洞见和实用技巧。
12 4
|
8天前
|
设计模式 缓存 测试技术
探索Python中的装饰器:从基础到高级应用
在本文中,我们将深入探讨Python中的装饰器,这是一种强大且灵活的工具,用于扩展或修改函数的行为。我们将从装饰器的基本概念和定义开始,逐步讲解它们的工作原理、如何创建和使用它们。接着,我们会探讨一些常见的装饰器用例,如日志记录、缓存和权限控制等。最后,本文将讨论一些高级话题,包括带参数的装饰器、使用functools模块增强装饰器以及装饰器与类方法的兼容问题。通过综合运用这些知识,您将能够更有效地利用Python的装饰器来优化您的代码。
21 10
|
1天前
|
监控 数据安全/隐私保护 Python
探索Python装饰器的本质与应用
本文深入探讨了Python中装饰器(Decorator)的工作原理、实际应用及其在软件开发中的重要性。通过浅显易懂的语言解释什么是装饰器,如何创建和运用装饰器来增强函数和类的功能。同时,文章还涵盖了一些高级主题,如带参数的装饰器、多层装饰以及装饰器的实际应用案例,帮助读者更全面地理解和掌握这一强大的编程工具。
6 1
|
6天前
|
前端开发 Python
Python编程的面向对象(二)—类的多态
Python编程的面向对象(二)—类的多态
13 7
|
5天前
|
IDE Java 开发工具
Python类与面向对象
Python类与面向对象
|
6天前
|
Python
? Python 装饰器入门:让代码更灵活和可维护
? Python 装饰器入门:让代码更灵活和可维护
12 4
|
6天前
|
缓存 测试技术 Python
探索Python中的装饰器:简化代码,提高可读性
【9月更文挑战第28天】在Python编程中,装饰器是一个强大的工具,它允许我们在不修改原有函数代码的情况下增加额外的功能。本文将深入探讨装饰器的概念、使用方法及其在实际项目中的应用,帮助读者理解并运用装饰器来优化和提升代码的效率与可读性。通过具体示例,我们将展示如何创建自定义装饰器以及如何利用它们简化日常的编程任务。
11 3
下一篇
无影云桌面