通过url_for 构造URL
首先我们要先了解url_for是做什么的函数。通过前面的学习我们已经知道我们可以通过一个URL映射到一个函数。反过来当我们知道这个函数的时候,想要获取它的URL的时候,这个时候我们就用到了这个函数url_for()
,url_for()
函数可以进行反转获取这个函数的URL。
url_for()
的函数的使用,函数可以接受一个及以上的参数,第一个接受的参数是你将要获取函数URL的函数名,如果你接受的这个函数支持传参,那么则可以传入url_for()
函数第一个参数的后面例如:url_for('article_list',aid=2)
。
示例:
from flask import Flask
from flask import url_for
app = Flask(__name__)
@app.route('/article/<aid>')
def article_list(aid):
return 'article %s' % aid
@app.route('/')
def index():
# 根据函数的名字进行反转,得到函数对应的路由
print(url_for('article_list',aid=2))
return 'Hello World'
if __name__ == '__main__':
app.debug = True
app.run(host='127.0.0.1',port=5000)
# app.run(debug=True) # 这一句等同于上面两行命令,app.run()默认参数:host='127.0.0.1',port=5000
运行服务访问:http://127.0.0.1:5000/,示例如下:
通过上面我们可以看到url_for()
方法获得了函数 article_list()
的路由,并显示在日志中。
函数有参数传参时,调用url_for
方法也需要进行传参。
为什么不在把 URL 写死在模板中,而要使用反转函数 url_for() 动态构建?
- 反转通常比硬编码 URL 的描述性更好。
- 您可以只在一个地方改变 URL ,而不用到处乱找。
- URL创建会为您处理特殊字符的转义,比较直观。
- 生产的路径总是绝对路径,可以避免相对路径产生副作用。
- 如果您的应用是放在 URL 根路径之外的地方(如在 /myapplication 中,不在 / 中), url_for() 会为您妥善处理。
示例:
from flask import Flask,url_for
app = Flask(__name__)
@app.route('/')
def index():
print(url_for('article_list',aid=2))
print(url_for('notice'))
print(url_for('follow', fid=2,page=5))
return 'Hello World'
@app.route('/article/<aid>')
def article_list(aid):
return 'article %s' % aid
@app.route('/notice')
def notice():
return 'Notice is as follows'
@app.route('/follows/<fid>')
def follow(fid):
return 'Follower %s' % fid
if __name__ == '__main__':
app.run(debug=True) # 另一种debug写法,详见上一篇文章
运行脚本,访问日志打印结果如下:
注:如果url_for()方法中提供的参数多于传入的函数的参数时,以查询参数的形式返回路由。
url_for()函数可以转义一些特殊字符和unicode字符串,示例 “/”转码测试:
from flask import Flask,url_for
app = Flask(__name__)
@app.route('/')
def index():
print(url_for('follow', fid=2,page=5,param='/'))
return 'Hello World'
@app.route('/follows/<fid>')
def follow(fid):
return 'Follower %s' % fid
if __name__ == '__main__':
app.run(debug=True)
运行结果日志显示:
注: / 被编码成ASCII码 %2F
唯一的 URL / 重定向行为
有些URL的末尾是有斜杠的,有些URL末尾是没有斜杠的,这其实是两个不同的URL。
@app.route('/projects/')
def projects():
return 'The project page'
@app.route('/about')
def about():
return 'The about page'
projects()
的 URL 是中规中矩的,尾部有一个斜杠,看起来就如同一个文件 夹。访问一个没有斜杠结尾的 URL ( /projects )时 Flask 会自动进行重 定向,帮您在尾部加上一个斜杠( /projects/ )。
about()
的 URL 没有尾部斜杠,因此其行为表现与一个文件类似。如果访问这 个 URL 时添加了尾部斜杠(/about/
)就会得到一个 404 “未找到” 错 误。这样可以保持 URL 唯一,并有助于搜索引擎重复索引同一页面。
建议: 在定义路由时加上末尾的/,这样在访问时加 / 与否都可以访问到。
指定HTTP方法
Web 应用使用不同的 HTTP 方法处理 URL 。当您使用 Flask 时,应当熟悉 HTTP 方法。 默认情况下定义的路由只能使用GET
请求。 如果想要使用其他的请求可以使用route()
装饰器的 methods
参数来处理不同的 HTTP 方法:
from flask import request
from flask import Flask
app = Flask(__name__)
@app.route('/article/<aid>')
def article_list(aid): # 默认支持GET请求
return 'article %s' % aid
@app.route('/login_0', methods=['GET']) #仅支持GET请求方法
def login_0():
if request.method == 'GET': # request.method 判断请求方法
return "这是一个GET请求"
else:
return "这是不一个GET请求"
@app.route('/login_', methods=['POST']) #仅支持POST请求方法
def login_():
if request.method == 'POST': # request.method 判断请求方法
return "这是一个POST请求"
else:
return "这是不一个POST请求"
@app.route('/login', methods=['GET', 'POST']) # 既能支持GET方法又能支持POST方法
def login():
if request.method == 'POST': # request.method 判断请求方法
return "这是一个POST请求"
else:
return "这是不一个POST请求"
if __name__ == '__main__':
app.run(debug=True)
示例如下:
可以发现装饰器route()
装饰器的 methods
参数让login
的URL既能支持GET方法又能支持POST方法,此时可以用POST方法请求到。
flask 参数传参
通过把 URL 的一部分标记为 <variable_name> 就可以在 URL 中添加变量。标记的 部分会作为关键字参数传递给函数。
from flask import Flask,url_for,request
app = Flask(__name__)
@app.route('/article/<aid>')
def article_list(aid):
return 'article %s' % aid
if __name__ == '__main__':
app.run(debug=True)
示例如下:
GET请求接收参数
request.args.get('xxx')
通过定义路由在@app.route()
中传入一个参数methods
参数来指定GET
请求。
from flask import Flask,url_for,request
app = Flask(__name__)
# 默认支持GET请求,可以不通过methods参数指定GET请求(建议指定)。
@app.route('/login/', methods=['GET'])
def login():
print(request.args.get('name'))
aa = request.args.get('name')
return 'login ' + aa
if __name__ == '__main__':
app.run(debug=True)
示例如下:
POST请求接收参数
request.form.get('xxx')
通过定义路由在@app.route()
中传入一个参数methods
参数来指定POST
请求。
from flask import Flask, url_for, request
app = Flask(__name__)
@app.route('/logout/', methods=[ 'POST'])
def logout():
print(request.form.get('name'))
aa = request.form.get('name')
return 'logout ' + aa
if __name__ == '__main__':
app.run(debug=True)
示例如下:
特别注意:
在给定了methods参数后,就只能用方法列表中的方法请求,如果列表中没有get方法则不能再用get方法进行请求。