目录
- 如何定义动态路由
- 注册动态路由
- 匹配动态路由
- 框架运行效果展示
- 总结
今天我们将继续对该框架进行路由添加正则表达式。
本篇文章所依赖的python环境为:

路由添加正则表达式有什么用?
在介绍之前,我们首先要介绍一下Get获取资源的时候,有如下几种传参方式:
- 使用
key...value的形式,将参数放到url问号?后面,如:http://127.0.0.1/userInfo?name=pdudo - 在
url中以路径参数的方式传递,例如:http://127.0.0.1/userInfo/pdudo/info。 - 在请求头中传递参数,请求头是以
key...value的形式存储的,当然也可以自定义存放请求参数。
上述的第二种以路径参数的方式传参,就是我们这边文章所描述的点。
为什么需要这样做呢?
假设我们现在正在写一个功能,他需要返回传上来的用户信息,接口为: /userInfo/userid/infos,其中userid是动态的,所以接口可以是/userInfo/c12345/infos,也可以是/userInfo/d33456/infos等等。
如果是这种情况,我们应该怎么样样定义路由呢? 是在项目中给每个人都定义一个路由信息么?
很显然不是的,如果这个时候,注册的路由恰恰好是正则的,可以获取客户端传上来的编程客栈参数,例如:/userInfo/pdudo/info,函数将会获取pdudo,这样不是很好么?
这就需要用到正则了。
如何定义动态路由
由于我们此前已经写好了定义静态路由的相关方法,所以我们在定义动态路由的时候,最好加一个识别参数,告诉框架,这个路由是静态的 还是 动态的。
由于是动态路由,所以在路由url中,有些值肯定是动态的,如何在区别于静态的值和动态的值呢? 在该框架中,我们将动态值用大括号括起来,比如: /userInfo/{userID}/infos,其中userID是动态的,可以是任何值组成,但是/userInfo和/infos必须是静态的,而且先后顺序也是固定的。
所以说,我们可以约束一下,在定义动态路由的时候,需要告诉框架,我这个是动态路由,以及动态url。
所以我们准备将动态路由定义规划如下:
@myWeb.routes(path="/jobs/{jobID}/startd",methods="post",regular=True)
def startJobs(r,cData):
print("cData: " ,cData)
上述代码,是我们即将完善功能后的代码案例,其中regular为True代表该路由是动态的,path的值为/jobs/{jobID}/startd,其中jobID也是动态的,可以被任何值代替。
最后是函数将接收2个参数,r的值代表wsgi中的environ信息。 而cData则代表从客户端上报的值,即:jobID的替代值。
注册动态路由
如上我们已经约束好了一个动态路由的添加规则,那么我们如何注册动态路由呢?
首先肯定的是,动态路由肯定不能和静态路由放在一起,所以我们定义了新的字典用于存放动态路由的信息,如:
mapGetRegularRouting = {}
mapPostRegularRouting = {}
而后则是我们需要将注册的路由信息转换为正则表达式编程,这里举个例子:
我们注册的路由是这样的: /jobs/{jobID}/startd,我们需要将其转换为正则表达式,正则表达式的值是这样的: ^/jobs/(.*?)/startd$。
关于这段正则表达式,它的含义是 匹配以/jobs/开头和/startd结尾的字符串,并且将匹配到的信息(.*的值)存储到元组中。
注意这里.*后面的问号?代表不启用贪婪匹配,也就是最小化匹配。
现在的问题是,如何将{jobsID}给转换为(.*?)呢?编程客栈 在python中,可以使用re.sub进行替换操作,我们可以使用如下语句,将{jobsID}给替换为(.*?),代码如下:
import re
print(re.sub("{.*?}","(.*?)","/jobs/{jobID}/startd"))
如上代码执行的结果为:

最后将该路由信息存储到上述定义好的字典中即可,代码片段如下:
reFindall = re.compile(r"{(.*?)}")
reSubAll = re.compile(r"{.*?}")
parameter = re.findall(reFindall,self.path)
reText = re.sub(reSubAll,"(.*?)",self.path)
reText = f"{reText}$"
regular = {
"original": self.path,
"reText": reText,
"parameter": parameter,
"func": func
}
# ...
if self.re:
mapPostRegularRouting[regular["reText"]] = regular
至此,我们的动态路由就已经注册好了。
匹配动态路由
相比于注册动态路由,当客户端请求来了之后,匹配动态路由会很麻烦,因为没有请求报文不是告诉你,这是动态路由还是静态路由,所以说,匹配动态路由很麻烦。
我们匹配动态路由想到的时候笨办法,即: 先匹配静态路由,若静态路由匹配不到,则再匹配动态路由,若动态路由都匹配不到的话,则选用默认路由。
匹配静态路由和动态路由,前面篇章已经介绍过了,所以这里不再赘述,这里就介绍如何匹配动态路由。
所谓的动态路由规则匹配,实际上就是拿着客户端上传的url,挨个对我们已有的正则表达式做轮训,若匹配成功则退出循环,若匹配失败,最后就返回一个默认路由即可,匹配正则表达式代码如下:
for key in RegularRouting.keys():
if re.match(key, path):
isRegular = True
collText = re.findall(key, path)
func = RegularRouting[key]["func"]
break
else:
func = Route["/*"]
框架运行效果展示
关于
myWeb.py由于很长,有100多行,不好截图,也不好纯复制代码了,所以放到了gitee上面:gitee.com/pdudo/golea…
我们简单写几个demo测试一下web框架,代码如下:
import myWeb
import wsgiref.simple_server
@myWeb.routes(path="/ip",methods="all")
def indx(r):
print(r["REMOTE_ADDR"])
return (200,r["REMOTE_ADDR"])
@myWeb.routes(path="/hello/开发者_开发学习{name}",methods="get",regular=True)
def helloWold(r,cData):
name = cData[0]
return (200,"hello %s" % (name))
def main():
s = wsgiref.simple_server.make_server('', 8888, myWeb.application)
s.serve_forever()
if __name__ == '__main__':
main()
上述代码,我们首编程客栈先引入了myWeb和wsgiref模块,前置是我们自己写的,后则是一个满足wsgi服务器框架,在代码中,我们定义了2个路由信息,一个是静态路由,一个是动态路由,其中静态路由的函数是indx,路由信息是/ip,匹配的客户端请求方法为get或者post,该方法主要返回ip地址。
动态路由的函数则是helloWold,它将匹配到以/hello/为首的路由信息,匹配客户端请求方法为get,还函数主要获取用户发送的动态值,并且以hello 等返回给客户端。
在主函数中,我们启动一个简单的wsgi服务器,入口为myWeb的application方法 。
代码运行效果如下:

我们分别使用get和post方法,请求本地/ip路由,可见都回复回来了,而后我们访问/hello/pdudo和/hello/juejin,他们也分别回复了hello pdudo和hello juejin。
总结
本篇文件介绍了将此前的web框架,路由修改为正则表达式形式,具体添加方式为: 在进行路由注册的时候,将其路由信息转换为正则表达式,而在进行路由匹配的时候,先匹配静态服务器,若静态服务器没有,再进行正则匹配,若正则匹配依然不行,则返回最后的默认页面。
以上就是python实现web应用框架之增加动态路由javascript的详细内容,更多关于python增加动态路由的资料请关注我们其它相关文章!
加载中,请稍侯......
精彩评论