flask模板的基本用法(定界符、模板语法、渲染模板),模板辅助工具(上下文、全局对象、过滤器、测试器、模板环境对象) --

flask模板

在动态web程序中,视图函数返回的HTML数据每每须要根据相应的变量(好比查询参数)动态生成。html

当HTML代码保存到单独的文件中时,咱们无法再使用字符串格式化或拼接字符串的当时在HTML代码中插入变量,这时咱们须要使用模板引擎(template engine)。借助模板引擎,咱们能够再HTML文件中使用特殊的语法来标记变量,这类包含固定内容和动态部分的可重用文件称为模板(template)。python

模板引擎的做用就是读取并执行模板中的特殊语法标记,并根据传入的数据将变量替换为实际值,输出最终的HTML页面,这个过程被称为渲染(rendering)。web

Flask默认使用的模板引擎是jinja2,他是一个功能齐全的python模板引擎,输了设置变量,还容许咱们在模板中添加if判断,执行for迭代,调整函数等,以各类方式 控制模板的输出。flask

对于jinja2来讲,模板能够是任何格式的纯文本文件,好比HTML、XML、CSV等。安全

模板的基本用法

下面介绍一下如何使用jinja建立HTML模板,并在视图函数中渲染模板,最终实现HTML响应的动态化session

建立模板

假设咱们须要编写一个用户的电影清单页面,模板中须要显示用户信息以及用户收藏的电影列表,包含电影的名字和年份。首先建立一些虚拟数据用于测试显示效果:数据结构

user = {
    'username': 'Grey Li',
    'bio': 'A boy who loves movies and music.'
}
movies = [
    {'name' : 'My Neighbor Totoro','year':'1988'},
    {'name': 'Three Colours trilogy', 'year': '1993'},
    {'name': 'Forrest Gump', 'year': '1994'},
    {'name': 'Perfect Blue', 'year': '1997'},
    {'name': 'The Matrix', 'year': '1999'},
    {'name': 'Memento', 'year': '2000'},
    {'name': 'The Bucket list', 'year': '2007'},
    {'name': 'Black Swan', 'year': '2010'},
    {'name': 'Gone Girl', 'year': '2014'},
    {'name': 'CoCo', 'year': '2017'}

 

咱们在templates目录下建立一个watchlist.html做为模板文件,而后使用jinja2支持的语法在模板中操做这些变量。app

template/watchlist.html:ide

 

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>{{ user.username }}'s Watchlist</title>
</head>
<body>
<a href = "{{ url_for('hello') }}">&larr; Return</a>
<h2>{{ user.username }}</h2>
{% if user.bio %}
    <i>{{ user.bio }}</i>
{% else %}
    <i>This user has not provided a bio.</i>
{% endif %}
{# 下面是电影清单(这是注释) #}
<h5>{{ user.username }}'s Watchlist ({{ movies[length] }}):</h5>
<ul>
    {% for movie in movies %}
        <li>{{ movie.name }} - {{ movie.year }}</li>
    {% endfor %}
</ul>
</body>
</html>

在模板中使用的&larr;是HTML实体,HTML实体除了用来转义HTML保留保留符号外,一般会被用来显示不容易经过键盘输入的字符。这里的&larr;会显示为左箭头,另外,&copy;用来显示版权标志。函数

在模板中添加python语句和表达式时,须要使用特定的定界符把他们标示出来。watchlist.html中设计的模板语法,咱们会在下面逐一介绍。首先,能够看到上面的代码中看到Jinja2里常见的三种定界符:

常见的三种定界符

 

一、  语句

好比if判断、for循环等:

{% … %}

二、  表达式

好比字符串、变量、函数调用等:

{{ … }}

三、  注释

{# … #}

另外,在模板中,Jinja2支持使用“.”获取变量的属性,好比user字典中的username键值经过“.”获取,即user.username,在效果上等同于user[‘username’]。

模板语法

利用jinja2这样的模板引擎,咱们能够将一部分的程序逻辑放到模板中去。简单地说,咱们能够在模板中使用python语句和表达式来操做数据的输出。但须要注意的是,jinja2并不支持全部python语法。而且出于效率和代码组织等方面的考虑,咱们应该适度使用模板,仅把和输出控制有关的逻辑操做放到模板中。

jinja2容许你在模板中使用大部分python对象,好比字符串、列表、字典、元组、整型、浮点型、布尔值。它支持基本的运算符号(+、-、*、/等)、比较符号(好比==、!=等)、逻辑符号(and、or、not和括号)以及in、is、None和布尔值(True、False)。

jinja2提供了多种控制结构来控制模板的输出,其中for和if是最经常使用的两种。jinja2里,语句使用{% … %}表示,尤为须要注意的是,在语句结束的地方,必须添加结束标签:

{% if user.bio %}
    <i>{{ user.bio }}</i>
{% else %}
    <i>This user has not provided a bio.</i>
{% endif %}

在这个if语句里,若是user.bio已经定义,就渲染{%if user.bio%}和{%else%}之间的内容,不然就渲染{%else%}和{%endif%}之间的默认内容。末尾的{%endif%}用来声明if语句的结束,这一行不能省略。

和python里同样,for语句用来迭代一个序列:

<ul>
    {% for movie in movies %}
        <li>{{ movie.name }} - {{ movie.year }}</li>>
    {% endfor %}
</ul>>

和其余语句同样,你须要在for循环的结尾使用endfor标签声明for语句的结束。在for循环内,jinja2提供了多个特殊变量,经常使用的for循环变量如图:

渲染模板

渲染一个模板,就是执行模板的代码,并传入全部在模板中使用的变量,渲染后的结果就是咱们要返回给客户端的HTML响应。在视图函数中渲染模板时,咱们并不直接使用jinja2提供的函数,而是使用flask提供的渲染函数render_template()

from flask import Flask,render_template
@app.route('/watchlist')
def watchlist():
    return render_template('watchlist.html',user=user,movies = movies)

在render_template()函数中,咱们首先传入的模板的文件名做为参数。Flask会在程序根目录下的templates文件夹里寻找模板文件,因此这里传入的文件路径是相对于templates根目录的。除了模板文件路径,咱们还以关键字参数的形式传入了模板中使用的变量值,以user为里:左边的user表示传入模板的变量名称,右边的user则是要传入的对象。

除了render_template()函数,Flask还提供了一个render_template_string()函数用来渲染模板字符串。

其余类型的变量经过相同的方式传入。传入jinja2中的变量值能够是字符串、列表和字典,也能够是函数、类和类实例,这彻底取决于你在视图函数传入的值。

例如:

<p>这时列表my_list的第一个元素:{{ my_list[0] }}</p>

<p>这是元祖my_tuple的第一个元素:{{ my_tutple[0] }}</p>

<p>这是字典my_dict的键为name的值:{{ my_dict[‘name’] }}</p>

<p>这是函数my_func的返回值:{{ my_func() }}</p>

<p>这是对象my_object调用某方法的返回值:{{ my_object.name() }}</p>

 

若是你想传入函数在模板中调用,那么须要传入函数对象自己,而不是函数调用(函数的返回值),因此近些出函数名称便可。当把函数传入模板后,咱们能够像在python脚本中同样经过添加括号的方式调用,并且你也能够在括号中传入参数。

根据咱们传入的虚拟数据,render_template()渲染后返回的HTML数据以下所示:

<!DOCTYPE html>

<html lang="en">

<head>

    <meta charset="UTF-8">

    <title>Grey Li's Watchlist</title>

</head>

<body>

<a href = "/hello">&larr; Return</a>

<h2>Grey Li</h2>

 

    <i>A boy who loves movies and music.</i>

 

 

<h5>Grey Li's Watchlist ():</h5>

<ul>

   

        <li>My Neighbor Totoro - 1988</li>

        <li>Three Colours trilogy - 1993</li>

        <li>Forrest Gump - 1994</li>

        <li>Perfect Blue - 1997</li>

        <li>The Matrix - 1999</li>

        <li>Memento - 2000</li>

        <li>The Bucket list - 2007</li>

        <li>Black Swan - 2010</li>

        <li>Gone Girl - 2014</li>

        <li>CoCo - 2017</li>

   

</ul>>

</body>

</html>

在和渲染前的模板文件对比时你会发现,原模板中全部的jinja语句、表达式、注释都会在执行后被移除,而全部的变量都会替换为对应的数据,访问127.0.0.7:5000/watchlist便可以看到渲染后的页面:

 

模板辅助工具

除了基本语法,jinja2还提供了许多方便的工具,这些工具可让你更方便的控制模板的输出。为了方便测试,咱们在示例程序的templates目录下建立了一个根页面模板index.html。返回主页的index视图和watchlist视图相似:

from flask import render_template

@app.route('/')
def index():
    return render_template('index.html')

上下文

模板上下文包含了不少变量,其中包含咱们调用render_template()函数时手动传入的变量以及flask默认传入的变量。

除了渲染时传入变量,也能够在模板中定义变量,使用set标签:

{% set navigation = [(‘/’, ‘Home’), (‘/about’, ‘About’)]%}

你也能够将一部分模板数据定义为变量,使用set和endset标签声明开始和结束:

{% set navigation %}
    <li><a href="/">Home</a></li>
    <li><a>href="/about"></a></li>
{% endset %}
内置上下文变量

Flask在模板上下文中提供了一些内置变量,能够在模板中直接使用

自定义上下文

若是多个模板都须要使用同一变量,那么比起在多个视图函数中重复传入,更好的办法是可以设置一个模板全局变量。flask提供了一个app.context_processor装饰器,能够用来注册模板上下文处理函数,它能够帮咱们完成统一传入变量的工做。模板上下文处理函数须要返回一个包含变量键值对的字典

注册模板上下文处理函数:

@app.context_processor
def inject_foo():
    foo = 'I am foo.'
   
return dict(foo=foo)#等同于return {'foo': foo}
当咱们调用render_remplate()函数渲染任意一个模板时,全部使用app.context_processor装饰器注册的模板上下文处理函数(包括flask内置的上下文处理函数)都会被执行,这些函数的返回值会被添加到模板中,所以咱们能够在模板中直接使用foo变量。
和在render_remplate()函数中传入变量相似,除了字符串、列表等数据结构,你也能够传入函数、类或类实例。
除了使用app.context_processor装饰器,也能够直接将其做为方法调用,传入模板上下文处理函数。
def inject_foo():
    foo = "I am foo."
   
return dict(foo=foo)

app.context_processor(inject_foo)
 
使用lambda能够简化为:
app.context_processor(lambda:dict(foo='I am foo.'))

全局对象

全局对象是指在全部的模板中均可以直接使用的对象,包括在模板中导入的模板

设置全局函数
jinja2在模板中默认提供了一些全局函数,经常使用的三个函数:

 

除了jinja2内置的全局函数,flask也在模板中内置了两个全局函数

 

 

flask除了把g、session、config、request对象注册上下文变量,也将他们设置为全局变量,所以能够全局使用。

url_for()用来获取URL,用法和在python脚本相同。在前面给出的watchlist.html模板中,用来返回主页的连接直接写出。在实际的代码中,这个URL使用url_for()生成,传入index视图的端点(index):

<a href=”{{ url_for(‘index’) }}”>&larr; Return</a>

 

自定义全局函数

除了使用app.context_processor注册模板上下文处理函数来传入变量,咱们也可使用app.template_global装饰器直接将函数注册为模板全局函数。好比,下面例子把bar()函数注册为模板全局函数。

@app.template_global()
def bar():
    return 'I am bar.'

默认使用函数的原名称传入模板,在app.template_global()装饰器中使用name参数能够指定一个自定义名称。app.template_global()仅能用于注册全局函数。

 

你能够直接使用app.add_template_global()方法注册自定义全局函数,传入函数对象和可选的自定义名称(name),好比app.add_template_global(your_global_function)。

 

过滤器

在jinja2中,过滤器(filter)是一些能够用来修改和过滤变量值的特殊函数,过滤器和变量用一个数显(管道符号)隔开,须要参数的过滤器能够像函数同样使用括号传递。

下面是一个堆name变量使用title过滤器的例子:

{{ name|title }}

这会将name变量的值标题化,至关于在python里调用name.title()。再好比,在前面的示例模板watchlist.html中使用length获取movies列表的长度,相似于在python中调用len(movies):

{{ movies|length }}

 

另外一种用法是将过滤器做用于一部分模板数据,使用filter标签和endfilter标签声明开始和结束。好比,下面是用upper过滤器讲一段文字转换为大写:

{% filter upper %}
    This text becomes uppercase.
{% endfilter % }
内置过滤器

jinja2提供了许多内置过滤器,经常使用的过滤器有:

 

 

在使用过滤器时,列表中过滤器函数的第一个参数表示被过滤的变量值(value)或字符串(s),即竖线符号左侧的值,其余的参数能够经过添加括号传入。

 

另外,过滤器能够叠加使用,下面的示例为name变量设置默认值,并将其标题化:

<h1>Hello, {{ name|default('陌生人')|title }}!</h1>

在以前学习XSS攻击的防范措施时,主要是对用户输入的文本进行转义,根据flask的设置,jinja2会自动对模板中的变量进行转义,因此咱们不用手动使用escape过滤器或调用escape()函数对变量进行转义。

默认的自动公开其转义仅针对.html、.htm、.xml以及.xtml后缀的文件,用于渲染模板字符串的render_template_string()函数也会对全部传入的字符串进行转义。

 

在确保变量值安全的状况下,若是你想避免转义,将变量做为HTML解析,能够对变量使用safe过滤器:

{{ santitized_text|safe }}

 

另外一种将文本记为安全的方法是在渲染前将变量转换为markup对象:

from fflask import Markup

@app.route('/hello6')
def hello6():
    text = Markup('<h>Hello, Flask!</h1>')
    return render_template('index.html',text=text)

这时在模板中能够直接使用{{ text }}

 

绝对不要直接对用户输入的内容使用safe过滤器,不然容易被植入恶意代码,致使XSS攻击。

 

自定义过滤器

若是内置的过滤器不能知足你的须要,还能够添加自定义过滤器。使用app.template_filter()装饰器能够注册自定义过滤器

例子:

from flask import Markup

@app.template_filter()
def musical(s):
    return s + Markup(' &#9835;')

和注册全局函数相似,你能够在app.template_filter()中使用name关键字设置过滤器的名称,默认会使用函数名称。过滤器函数须要接收被处理的值做为输入,返回处理后的值。过滤器函数接收s做为过滤的变量值,返回处理后的值。咱们建立的musical过滤器会在被过滤的变量字符后边添加一个音符图标,由于音符经过HTML实体&#9835;表示,咱们使用Markup类将它标记为圈圈字符。在使用时和其余过滤器用法相同:

{{ name|musical }}

 

你可直接一个app.add_template_filter()方法注册自定义过滤器,传入函数对象和可选的自定义名称(name),好比app.add_template_filter(your_filer_function)。

 

测试器

在jinja2中,测试器(Test)是一些用来测试变量或表达式,返回布尔值(True或False)的特殊函数。好比,number测试器用来判断一个变量或表达式是不是数字,咱们使用is链接变量和测试器:

{% if age is number %}
    {{ age * 365 }}
{% else %}
    无效的数字
{% endif %}
内置测试器

jinja2内置了许多测试器,经常使用的测试器及用法说明以下表:

 

在使用测试器时,is的左侧是测试器函数的第一个参数(value),其余参数能够添加括号传入,也能够在右侧使用空格链接,以sames为例:

{% if foo is sameas(bar) %}…

等同于:

{% if foo is samesas bar %}…

自定义测试器

和过滤器相似,咱们可使用flask提供的app.template_test()装饰器来注册一个自定义测试器。下例中,咱们建立一个baz过滤器,用来验证被测值是否为baz:

@app.template_test()
def baz(n):
    if n == 'baz':
        return True
    return False

测试器的名称默认为函数名称,你能够在app.template_test()中使用name关键字指定自定义名称。测试器函数须要接收被测试的值做为输入,返回布尔值。

你能够直接使用app.add_template_test()方法注册自定义测试器,传入函数对象和可选的自定义名称(name),好比app.add_template_test(your_test_function)。

模板环境对象

在jinja2中,渲染行为由jinja2.Environment类控制,全部的配置选项、上下文变量、全局函数、过滤器和测试器都存储在Environment实例上。当于flask结合后,咱们并不单首创建Environment对象,而是使用flask建立的Environment对象,它存储在app.jinja_env属性上。

在程序中,咱们可使用app.jinja_env更改jinja2设置。好比,能够自定义全部的定界符。下面使用variable_start_string和variable_end_string分别自定义变量定界符的开始和结束符号:

app = Flask(__name__)

app.jinja_env.variable_start_string = ’[[’

app.jinja_env.variable_end_string = ‘}}’

在实际开发中,若是修改jinja2的定界符,那么须要注意与扩展提供模板的兼容问题,通常不建议修改。

 

模板环境中的全局函数、过滤器和测试器分别存储在Environment对象的globals、filters和tests属性中,这三个属性都是字典对象。除了使用flask提供的装饰器和方法注册自定义函数,咱们也能够直接操做这三个字典添加相应的函数或变量,这经过向对应的字典属性中添加一个键值对实现,把名称做为键传入模板,对应的函数对象或变量做为值。下面是几个简单的例子。

添加自定义全局对象

使用app.jinja_env.globals分别向模板中添加全局函数bar和局部变量foo:

def bar():
    return "I am bar"
foo = "i am foo"
app.jinja_env.globals['bar'] = bar
app.jinja_env.globals['foo'] = foo

 

添加自定义过滤器

使用app.jinja_env.filters向模板中添加自定义过滤器smiling:

def smiling(s):
    return s + ' :'
app.jinja_env.filters['smiling'] = smiling
添加自定义测试器

使用app.jinja_env.tests向模板中添加自定义测试器baz

def baz(n):
    if n == 'baz':
        return True
    return False
app.jinja_env.tests['baz'] = baz
相关文章
相关标签/搜索