URL与视图

URL与函数的映射:

从之前的helloworld.py文件中,我们已经看到,一个URL要与执行函数进行映射,使用的是@app.route装饰器。@app.route装饰器中,可以指定URL的规则来进行更加详细的映射,比如现在要映射一个文章详情的URL,文章详情的URL/article/id/,id有可能为1、2、3…,那么可以通过以下方式:

  1. @app.route('/article/<id>/')
  2. def article(id):
  3. return '%s article detail' % id

其中<id>,尖括号是固定写法,语法为<variable_name>variable_name默认的数据类型是字符串。如果需要指定类型,则要写成<converter:variable_name>,其中converter就是类型名称,可以有以下几种:

  • string: 默认的数据类型,接受没有任何斜杠“\/”的文本。
  • int: 接受整形。
  • float: 接受浮点类型。
  • path: 和string的类似,但是接受斜杠。
  • uuid: 只接受uuid字符串。
  • any:可以指定多种路径,这个通过一个例子来进行说明:

    1. @app.route('/<any(article,blog):url_path>/')
    2. def item(url_path):
    3. return url_path
  • 以上例子中,item这个函数可以接受两个URL,一个是/article/,另一个是/blog/。并且,一定要传url_path参数,当然这个url_path的名称可以随便。

如果不想定制子路径来传递参数,也可以通过传统的?=的形式来传递参数,例如:/article?id=xxx,这种情况下,可以通过request.args.get('id')来获取id的值。如果是post方法,则可以通过request.form.get('id')来进行获取。

构造URL(url_for):

一般我们通过一个URL就可以执行到某一个函数。如果反过来,我们知道一个函数,怎么去获得这个URL呢?url_for函数就可以帮我们实现这个功能。url_for()函数接收两个及以上的参数,他接收函数名作为第一个参数,接收对应URL规则的命名参数,如果还出现其他的参数,则会添加到URL的后面作为查询参数
通过构建URL的方式而选择直接在代码中拼URL的原因有两点:

  1. 将来如果修改了URL,但没有修改该URL对应的函数名,就不用到处去替换URL了。
  2. url_for()函数会转义特殊字符和Unicode数据,这些工作都不需要我们自己处理。

下面用一个例子来进行解释:

  1. from flask import Flask,url_for
  2. app = Flask(__name__)
  3. @app.route('/article/<id>/')
  4. def article(id):
  5. return '%s article detail' % id
  6. # 这行的代码可以在交互模式下产生请求上下文,不用`app.run()`来运行这个项目,直接可以运行下面的代码,
  7. # 也会有`flask`上下文
  8. with app.test_request_context():
  9. print url_for('article',id='1')
  10. print url_for('article',id='2',next='/')
  1. 执行后的结果如下:
  2. > /article/1/
  3. > /article/2/?next=%2F

自定义URL转换器:

刚刚在URL映射的时候,我们看到了Flask内置了几种数据类型的转换器,比如有int/string等。如果Flask内置的转换器不能满足你的需求,此时你可以自定义转换器。自定义转换器,需要满足以下几个条件:

  1. 转换器是一个类,且必须继承自werkzeug.routing.BaseConverter
  2. 在转换器类中,实现to_python(self,value)方法,这个方法的返回值,将会传递到view函数中作为参数。
  3. 在转换器类中,实现to_url(self,values)方法,这个方法的返回值,将会在调用url_for函数的时候生成符合要求的URL形式。

比如,拿一个官方的例子来说,Reddit可以通过在URL中用一个加号(+)隔开社区的名字,方便同时查看来自多个社区的帖子。比如访问“www.reddit.com\/r\/flask+lisp\/”的时候,就同时可以查看flask和lisp两个社区的帖子,现在我们自定义一个转换器来实现这个功能:

  1. #coding: utf-8
  2. from flask import Flask,url_for
  3. from werkzeug.routing import BaseConverter
  4. class ListConverter(BaseConverter):
  5. def __init__(self,url_map,separator='+'):
  6. super(ListConverter,self).__init__(url_map)
  7. self.separator = separator
  8. def to_python(self, value):
  9. return value.split(self.separator)
  10. def to_url(self, values):
  11. return self.separator.join(BaseConverter.to_url(self,value) for value in values)
  12. app.url_map.converters['list'] = ListConverter
  13. @app.route('/community1/<list:page_names>')
  14. def community1(page_names):
  15. return '%s+%s' % tuple(page_names)
  16. @app.route('/community2/<list('|'):page_names>/')
  17. def community2(page_names):
  18. return "%s|%s" % tuple(page_names)

communityu1使用的是默认的+号进行连接,而第二种方式使用了|进行连接。

URL唯一:

FlaskURL规则是基于Werkzeug的路由模块。这个模块的思想是基于Apache以及更早的HTTP服务器的主张,希望保证优雅且唯一的URL
举个例子:

  1. @app.route('/projects/')
  2. def projects():
  3. return 'project page'

上述例子中,当访问一个结尾不带斜线的URL会被重定向到带斜线的URL上去。这样有助于避免搜索引擎搜索同一个页面两次。
再看一个例子:

  1. @app.route('/about')
  2. def about():
  3. return 'about page'

以上例子中,当访问带斜线的URL(\/about\/)会产生一个404(”Not Found”)错误。

指定HTTP方法:

@app.route()中可以传入一个关键字参数methods来指定本方法支持的HTTP方法,默认只响应GET请求,看以下例子:

  1. @app.route('/login/',methods=['GET','POST'])
  2. def login():
  3. return 'login'

以上装饰器将让loginURL既能支持GET又能支持POST

页面跳转和重定向:

重定向分为永久性重定向和暂时性重定向,在页面上体现的操作就是浏览器会从一个页面自动跳转到另外一个页面。比如用户访问了一个需要权限的页面,但是该用户当前并没有登录,因此我们应该给他重定向到登录页面。

  • 永久性重定向:http的状态码是301,多用于旧网址被废弃了要转到一个新的网址确保用户的访问,最经典的就是京东网站,你输入www.jingdong.com的时候,会被重定向到www.jd.com,因为jingdong.com这个网址已经被废弃了,被改成jd.com,所以这种情况下应该用永久重定向。
  • 暂时性重定向:http的状态码是302,表示页面的暂时性跳转。比如访问一个需要权限的网址,如果当前用户没有登录,应该重定向到登录页面,这种情况下,应该用暂时性重定向。

flask中,重定向是通过flask.redirect(location,code=302)这个函数来实现的,location表示需要重定向到的URL,应该配合之前讲的url_for()函数来使用,code表示采用哪个重定向,默认是302也即暂时性重定向,可以修改成301来实现永久性重定向。
以下来看一个例子,关于在flask中怎么使用重定向:

  1. from flask import Flask,url_for,redirect
  2. app = Flask(__name__)
  3. app.debug = True
  4. @app.route('/login/',methods=['GET','POST'])
  5. def login():
  6. return 'login page'
  7. @app.route('/profile/',methods=['GET','POST'])
  8. def profile():
  9. name = request.args.get('name')
  10. if not name:
  11. # 如果没有name,说明没有登录,重定向到登录页面
  12. return redirect(url_for('login'))
  13. else:
  14. return name