Python的Flask框架标配模板引擎Jinja2的使用教程

作者:邪恶的Sheldon 时间:2022-07-17 08:47:35 

Jinja2需要Python2.4以上的版本。
安装
按照Jinja有多种方式,你可以根据需要选择不同的按照方式。
使用easy_install 或pip:


#sudo easy_install Jinja2
#sudo pip install Jinja2

这两个工具可以自动从网站上下载Jinja,并安装到python目录的site-packages目录中。
从tar包安装:


# 下载Jinja的安装包
# 解压缩
# sudo python setup.py install

基本API用法
用Jinja创建模板最简单的方式是通过 Template. 但在实际应用中并不推荐此用法: 


<pre>
>>> from Jinja2 import Template
>>> template = Template('Hello {{ name }}!')
>>> template.render(name='World')
u'Hello World!'
</pre>

这个例子使用字符串作为模板内容创建了一个Template实例,然后用"name='World'"作为参数调用"render方法,将内容中 的'name'替换为"World",最终返回渲染过的字符串--"u'Hello World!'"。
有两种分隔符。{% raw %}{% ... %}{% endraw %} 和 {% raw %}{{ ... }}{% endraw %}。第一个用于执行类似 for 循环或者赋值的声明,后者是用于输出表达的结果到模板中。

如何组织模板
那么模板如何融入到我们的应用程序?如果你一直关注 Flask 的话,你可能注意到了 Flask 是十分灵活,它并没有对其内容进行一些特殊的限制。模板也不例外。你可能也注意到了通常有一个推荐的地方来放置东西(比如,模板)。对于模板而言,那个地方就是在包的目录里。


myapp/
 __init__.py
 models.py
 views/
 templates/
 static/
run.py
requirements.txt
templates/
 layout.html
 index.html
 about.html
 profile/
   layout.html
   index.html
 photos.html
 admin/
   layout.html
   index.html
   analytics.html

templates 目录的结构是与我们路由结构平行的。对于路由 myapp.com/admin/analytics 的模板就是 templates/admin/analytics.html。在目录里面还有一些额外的模板,它们不会直接地被渲染。layout.html 文件是为了让其它的模板继承。

继承
很像蝙蝠侠的背景故事一样,一个组织优秀的模板目录很大程度上依靠继承。父模板 通常定义一个通用的结构,所有 子模板 都能很好的继承它。在我们的例子中,layout.html 就是一个父模板而其它 .html 文件就是子模板。
你通常有一个顶层的 layout.html,它定义了你的应用程序的通用布局以及你的网站的每一部分。如果你看看上面的目录的话,你会看到一个顶层的 myapp/templates/layout.html,同样还有 myapp/templates/profile/layout.html 和 myapp/templates/admin/layout.html。最后两个文件继承和修改第一个文件。


{# _myapp/templates/layout.html_ #}
<!DOCTYPE html>
<html lang="en">
 <head>
   <title>{% raw %}{% block title %}{% endblock %}{% endraw %}</title>
 </head>
 <body>
 {% block body %}
   <h1>This heading is defined in the parent.</h1>
 {% endblock %}
 </body>
</html>

在子模板中,我们可以扩展父模板并且定义这些块的内容。


{# _myapp/templates/index.html_ #}
{% extends "layout.html" %}
{% block title %}Hello world!{% endblock %}
{% block body %}
 {{ super() }}
 <h2>This heading is defined in the child.</h2>
{% endblock %}

super() 函数让我们渲染父级块的内容。

创建宏
我们可以在我们模板中坚持 DRY(不要重复自己)的原则,通过抽象出重复出现的代码片段到 宏。如果我们正工作在为我们应用程序导航的 HTML 上,我们需要给一个 “活跃的”链接一个 class(class=”active”)。没有宏的话,我们要编写一大段 if ... else 语句,这些语句检查每一个链接找到正处于活跃的一个。
宏提供了一种模块化代码的方式;它们像函数一样工作。让我们看看如何使用宏标记一个活跃的链接。


{# myapp/templates/layout.html #}
{% from "macros.html" import nav_link with context %}
<!DOCTYPE html>
<html lang="en">
 <head>
 {% block head %}
   <title>My application</title>
 {% endblock %}
 </head>
 <body>
   <ul class="nav-list">
     {{ nav_link('home', 'Home') }}
     {{ nav_link('about', 'About') }}
     {{ nav_link('contact', 'Get in touch') }}
   </ul>
 {% block body %}
 {% endblock %}
 </body>
</html>

在这个模板中我们现在要做的就是调用一个未定义的宏 - nav_link -接着向其传递两个参数:目标端点(例如,目标视图的函数名)以及我们要显示的文本。
你可能会注意到在导入语句中我们指定了 with context。Jinja 的 context 是由传递到 render_template() 函数的参数以及来自我们的 Python 代码的 Jinja 环境上下文组成。对于模板来说,这些变量在模板被渲染的时候是可用的。
一些变量是明显地由我们传入,例如,render_template("index.html", color="red"),但是还有一些变量和函数是由 Flask 自动地包含在上下文中,例如,request, g 和 session。当我们说 {% raw %}{% from ... import ... with context %}{% endraw %} 的时候,就是告诉 Jinja 这些变量对宏也可用。
现在是时候定义在我们模板中使用的 nav_link 宏。


{# myapp/templates/macros.html #}
{% macro nav_link(endpoint, text) %}
{% if request.endpoint.endswith(endpoint) %}
 <li class="active"><a href="{{ url_for(endpoint) }}">{{text}}</a></li>
{% else %}
 <li><a href="{{ url_for(endpoint) }}">{{text}}</a></li>
{% endif %}
{% endmacro %}

现在我们已经在 myapp/templates/macros.html 中定义了宏。在这个宏中我们使用了 Flask 的 request 对象 — 默认情况下在 Jinja 上下文中是可用的 — 用来检查传入到 nav_link 中的路由的端点是否是当前请求。如果是,我们正在当前页面上,接着我们标记它为活跃的。
从 x 导入 y 语句采用了 x 的相对路径。如果我们的模板是 myapp/templates/user/blog.html,我们可以在使用 from "../macros.html" 导入 nav_link。

自定义过滤器
Jinja 过滤器是一个函数,它能够在 {% raw %}{{ ... }}{% endraw %} 中用于处理一个表达式的结果。在表达式结果输出到模板之前它就被调用。


<h2>{{ article.title|title }}</h2>

在这段代码中,title 过滤器接收 article.title 作为参数并且返回一个过滤后的标题,接着过滤后的标题将会输出到模板中。这就像 UNIX 的“管道化”一个程序到另一个程序的输出。
有很多像 title 一样的内置过滤器。请参阅 Jinja 文档中的 完整列表。
我们可以在我们的 Jinja 模板中定义自己的过滤器供使用。举例来说,我们将会实现一个简单 caps 过滤器用来大写一个字符串中所有的字母。
Jinja 已经有一个 upper 过滤器来做这样的事情,并且还有一个 capitalize 过滤器,它能用来大写第一个字母,小写其余的字母。这些也能处理 unicode 转换,但是我们会继续我们的示例,让大家目前能够知道如何自定义过滤器。
我们要在 myapp/util/filters.py 中定义我们的过滤器。这里给出一个 util 包,它里面有各种各样的模块。


# myapp/util/filters.py
from .. import app
@app.template_filter()
def caps(text):
 """Convert a string to all caps."""
 return text.uppercase()

在这段代码中我们使用 @app.template_filter() 装饰器注册我们的函数成一个 Jinja 过滤器。默认的过滤器名称就是函数的名称,但是你可以传入一个参数到装饰器中来改变它。


@app.template_filter('make_caps')
def caps(text):
 """Convert a string to all caps."""
 return text.uppercase()

现在我们可以在模板中调用 make_caps 而不是 {% raw %}caps:{{ "hello world!"|make_caps }}{% endraw %}。
为了要让我们的过滤器在模板中可用的话,我们只需要在我们的顶层 \\_init.py\\_ 的中导入它。


# myapp/__init__.py
# Make sure app has been initialized first to prevent circular imports.
from .util import filters
标签:Python,Jinja2
0
投稿

猜你喜欢

  • 基于HTML+JS实现简单的年龄计算器

    2024-04-23 09:27:29
  • MySQL索引类型一览 让MySQL高效运行起来

    2010-04-22 16:52:00
  • window.top[_CACHE]实现多个jsp页面共享一个js对象

    2023-07-18 09:25:14
  • PHP延迟静态绑定使用方法实例解析

    2024-06-05 15:41:45
  • 学会python自动收发邮件 代替你问候女友

    2023-07-08 17:01:49
  • python 使用值来排序一个字典的方法

    2022-02-05 00:25:05
  • Pandas时间数据处理详细教程

    2022-04-29 03:14:22
  • Python 连接字符串(join %)

    2021-01-13 23:30:46
  • Python实现的多线程端口扫描工具分享

    2023-03-30 04:32:19
  • js实现简单的联动菜单效果

    2024-04-19 09:54:45
  • asp显示左边的n个字符自动识别汉字的函数

    2007-09-13 12:16:00
  • PHP实现通过正则表达式替换回调的内容标签

    2024-05-13 09:25:39
  • js实现简单选项卡功能

    2024-04-22 13:05:47
  • Python实例练习水仙花数问题讲解

    2023-05-26 18:38:11
  • asp,php,.net使用301重定向方法

    2007-09-26 14:05:00
  • Python源码加密与Pytorch模型加密分别介绍

    2023-11-30 13:05:03
  • 仿谷歌主页js动画效果实现代码

    2024-04-17 10:07:36
  • 网页对联广告代码效果大全

    2007-10-25 23:16:00
  • vue 开发一个按钮组件的示例代码

    2024-04-30 10:27:40
  • python中显存回收问题解决方法

    2022-06-28 03:06:29
  • asp之家 网络编程 m.aspxhome.com