Flask中的路由
查看整个flask中的路由映射关系 app.url_map
from flask import Flask
app = Flask(__name__)
@app.route("/index")
def index()
return "flask index page"
if __name__ == "__main__":
print(app.url_map)
app.run()
# 下面是打印效果
'''
Map([<Rule '/index' (OPTIONS, HEAD, GET) -> index>,
<Rule '/static/<filename>' (OPTIONS, HEAD, GET) -> static>])
'''
打印的效果是一个列表,里面包含着一个一个路由规则,拿出一条解析<Rule '/index' (OPTIONS, HEAD, GET) -> index>
rule
: 规则的意思
'/index'
: 这个就是访问的路径,也就是url
(OPTIONS, HEAD, GET)
: 允许的请求方式
index
: 对应的视图函数
代码块一:
@app.route("/index")
def index()
return "flask index page"
按照这样的例子来写,我们就实现了一个url和视图函数的对应关系,去看看route()这个方式都做了什么
代码块二:
def route(self, rule, **options):
def decorator(f):
endpoint = options.pop('endpoint', None)
self.add_url_rule(rule, endpoint, f, **options)
return f
return decorator
首先看这行代码 @app.route("/index")
route
它是一个方法对吧,那么route("/index")
它加了括号,是不是表示在我们执行falsk程序的时候,就会调用,那它的返回值就应该就是一个函数的内存地址,也就是源码中的decorator
函数的内存地址,那么代码块一
不是可以更换为这样的形式
代码块三
@decorator
def index():
return "flask index page"
继续看这段代码,一个装饰器对吧,会将@decorator
下一行的函数(index
)的内置地址当作参数传递到函数中,看代码块二
中的decorator
函数都做了一些什么,decorator
函数最终还是返回了参数f
,这个参数f
就是代码块三
中的index
函数对吧,那在retuen之前还做了什么呢?也就是这两行代码endpoint = options.pop('endpoint', None)
, self.add_url_rule(rule, endpoint, f, **options)
, 那我们继续分析吧
endpoint = options.pop('endpoint', None)
这行代码中的options
就是一个字典,使用了pop方法,那么这个字典是怎么得到的呢?看代码块二
那段源码,有一个**options
它和**kwargs
是一个意思,是一个字典。那么route
这个方法的参数传递方式应该是这样了 route(rule,k1=v1,k2=v2)
self.add_url_rule(rule, endpoint, f, **options)
这行代码才是最主要的地方,就是这行代码将视图函数和url之前相互对应起来。它的参数我应该不用再细说了,我们看add_url_rule
方法做了什么
算了,源码我就不讲了,有兴趣的话可以自己去阅读下,这个方法主要就是帮我们在url和视图函数之间做了对应关系,那么decorator
这个函数也就是代码二
,它没有对我们的视图函数做处理,就返回了,也就多执行了add_url_rule
这个方法对吧
# 为创建视图函数和url之间的对应关系的主要代码就是 add_url_rule
# 那么我们除了代码块一那种方式创建关系,下面这段代码也可以达到效果
from flask import Flask
app = Flask(__name__)
def index()
return "flask index page"
app.add_url_rule("/",view_func=index)
if __name__ == '__main__':
app.run()
# 这样的格式呢,在方法add_url_rule的注释中也提到了
好了,对于通过使用装饰器实现url和视图函数的对应关系的分析就结束了
我们回到app.url_map的的打印结果来看,里面一个允许求情的方式
@app.route('/index')
def index ():
return "index page"
这段代码我根本没有指定允许请求的方式,那么它允许的方式默认为GET方式,这时候向这个url发起POST请求,会报错,说该请求方式不允许
为视图函数设置允许请求方式:app.route('/index',methods=['GET','POST'])
,methods
它是一个列表,请求方法不区分大小写,他会同一为你upper(),
我们在看decorator
这个函数,也就是代码块二
中的内容
def decorator(f):
endpoint = options.pop('endpoint', None)
self.add_url_rule(rule, endpoint, f, **options)
return f
其中有一个这个endpoint
, 它代表什么呢?它就是为视图函数命别名,如果你不传的话,那么endpoint
的值就是None,最终在add_url_rule
会进行判断endpoint
是否为true,不是的话,会将视图函数的__name__
的值传递给endpoint
。最终我们可以通过反向查找endpoint的值
去找到对应的url。
url_for
这个就是反向查找的方法,通过 from flask import url_for
去导入,下面写一段代码演示下
from flask import Flask,url_for,redirect
# redirect是页面重定向的,后面我会继续更新
app = Flask(__name__)
@app.route("/register",endpoint='register')
def register():
'''访问register之后,重定向到login页面'''
url = url_for(login)
return redirect(url)
@app.route('/login',endpoint='login')
def login():
return "login page"
if __name__ == "__main__":
app.run()
使用endpoint
的好处是:不管/login
怎么变,使用url_for(login)
就能拿到登陆页面的url。
看这个url
为 /goods
, 这是一个获取商品页面的url,那我想获取某一个商品的页面,这个url怎么设计呢?
- ulr?goods_id = 1 这样的方式请求,实现起来很容易
- /goods/1 这里的1就是商品的id,那么是多变的对吧,所以必须找到设计出这样的url,/goods/商品的id
这里需要用到flask提供的转换器:
default
string
any
path
int
float
uuid
就这几个自带的
@app.route('/goods/<int:goods_id>',methods=["GET"])
def goods(goods_id):
print(goods_id)
return "goods page {}".format(goods_id)
# 注意goods函数必须传递参数,参数名要与转换器后面的值一样,比如这里的goods_id
# 上面的几个转换器可以去自行了解下,看名字大概就知道怎么使用了
# 除了flask自带的转换器,我们也可以自定义转换器。
# 这时别人写的博客,可以去了解:https://www.cnblogs.com/serpent/p/9575179.html