在上一阶段,我们一起学习了基于Python地 web框架Flask,并且初步了解了这个框架有一种渲染方式叫做 模板语法,今天,我们一起再来深入地了解和学习这个叫做Jinja2地模板语法。
WEB开发 - Flask 入门:由浅入深地带你学习
Flask 的模板渲染由Jinja2提供支持,Jinja2 是一个强大的 Python 模板引擎。现在假设你已经对它有了基本的了解,所以接下来让我们更深入地了解 Flask 模板渲染的高级功能和最佳实践。
1. Jinja2 模板语法回顾
在进一步介绍之前,这里先简单概述一下 Jinja2 的语法:
- 变量:
{{ variable }}
- 控制结构:
{% if condition %} ... {% endif %}
,{% for item in items %} ... {% endfor %}
- 筛选器:
{{ variable|filter_name }}
- 评论:
{# This is a comment #}
- 继承:
{% extends "base.html" %}
,{% block content %} ... {% endblock %}
根据上面的模板语法,我们使用一个例程来实现Jinja2 模板语法回顾:
代码示例:
1.1 变量
在模板中,我们可以通过 {{ variable }}
来渲染变量的值。
示例:
<!DOCTYPE html>
<html lang="zh">
<head><meta charset="UTF-8"><title>Jinja2 模板语法回顾</title>
</head>
<body><h1>欢迎,{{ name }}!</h1><p>你当前的年龄是:{{ age }}岁。</p>
</body>
</html>
1.2 控制结构
使用 {% %}
来执行控制结构,例如条件判断和循环。
条件判断:
{% if age >= 18 %}<p>你是成年人。</p>
{% else %}<p>你是未成年人。</p>
{% endif %}
循环:
<ul>
{% for item in items %}<li>{{ item }}</li>
{% endfor %}
</ul>
1.3. 过滤器
过滤器用于修改输出的值,比如格式化日期、文本转换等。使用 |
来调用过滤器。
示例:
<p>{{ "2024-12-27"|date("yyyy年MM月dd日") }}</p>
<p>{{ "Hello World"|lower }}</p>
1.4. 模板继承
可以通过 {% extends %}
和 {% block %}
实现模板的继承和重用。
基本模板(base.html):
<!DOCTYPE html>
<html lang="zh">
<head><meta charset="UTF-8"><title>{% block title %}我的网站{% endblock %}</title>
</head>
<body><header><h1>{% block header %}欢迎来到我的网站{% endblock %}</h1></header><div class="content">{% block content %}内容区域{% endblock %}</div><footer>{% block footer %}版权信息{% endblock %}</footer>
</body>
</html>
上面的基本模版在浏览器中的显示:
子模板(home.html):
{% extends "base.html" %}{% block title %}首页 - 我的网站{% endblock %}{% block content %}<h2>欢迎来到首页!</h2><p>这里是一些重要的内容。</p>
{% endblock %}
上面的子模版在继承了基本模板之后,在浏览器中的显示(内容区域已经被子模版的内容替换掉了):
1.5. 自定义过滤器
你可以定义自定义的过滤器来对模板中的变量进行处理。
Python 代码(app.py):
from flask import Flask
from flask import render_templateapp = Flask(__name__)# 自定义过滤器函数
def reverse_string(value):return value[::-1]# 注册自定义过滤器
app.jinja_env.filters['reverse'] = reverse_string@app.route('/')
def home():return render_template('home.html')if __name__ == '__main__':app.run(debug=True)
模板代码(home.html):
<p>{{ "hello"|reverse }}</p> <!-- 输出:olleh -->
上面的webServer 运行后,在浏览器总访问效果如下:
你会发现,输入内容“hello” 被我们自己定义的反相器reverse给倒序了。这个反相器就是一个过滤器,你可以定义很多不同的过滤器,比如字符串截取、大小写变换、日期格式转换、字符串替换等等。
1.6. 模板宏
宏类似于模板中的函数,可以用来复用代码块。
示例:(macros.html)
{% macro render_item(item) %}<div class="item"><h3>{{ item.name }}</h3><p>{{ item.description }}</p></div>
{% endmacro %}
使用宏:(home.html)
{% from "macros.html" import render_item %}<div class="items">{% for item in items %}{{ render_item(item) }}{% endfor %}
</div>
app.js 如下
from flask import Flask, render_template# 创建一个 Flask 实例
app = Flask(__name__)@app.route('/home')
def greet():# 定义要传递给模板的数据items = [{"name": "Alice", "description": "description Alice"},{"name": "Tom", "description": "description Tom"}]# 将数据传递给模板return render_template('home.html', items=items)# 运行应用
if __name__ == '__main__':app.run(debug=True)
上述代码执行后,从浏览器访问:
你会发现,模板宏被调用了,显示了名字和介绍。这个模板宏,可以被很多具有相同数据结构的页面调用。它形成了一个类似vue的 component 组件。
1.7. 模板包含
{% include %}
用于包含其他模板的内容,通常用于包含页面的共同部分。
示例:
<div class="sidebar">{% include "sidebar.html" %}
</div>
通过这些语法,Jinja2 提供了灵活而强大的模板引擎,能帮助我们高效地渲染动态内容,并且让页面结构更清晰、可维护。
2.模板继承(高级)
模板继承是 Jinja2 最强大的功能之一。它允许您定义一个基本模板并在子模板中扩展它,从而实现通用布局和结构的重用。
基本模板示例:
<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><title>{% block title %}My Website{% endblock %}</title>
</head>
<body><header><h1>{% block header %}Welcome to My Website{% endblock %}</h1></header><div class="content">{% block content %}Content goes here{% endblock %}</div><footer>{% block footer %}Footer content{% endblock %}</footer>
</body>
</html>
子模板示例:
{% extends "base.html" %}{% block title %}Home - My Website{% endblock %}{% block content %}<h2>Welcome to the home page!</h2><p>Here is some important content.</p>
{% endblock %}
这个案例和我们之前回顾的模板调用实际上一样的,在此不做赘述了,需要加深理解的可以重做一遍。
3.动态模板渲染
您可以将数据动态传递给模板。将数据操作逻辑与 Flask 视图中的渲染逻辑分开是一种很好的做法。
Flask 路线示例:
from flask import Flask, render_templateapp = Flask(__name__)@app.route('/')
def home():data = {'name': 'John Doe','age': 30,'items': ['apple', 'banana', 'cherry']}return render_template('home.html', data=data)
模板示例(home.html
):
<h1>Hello, {{ data.name }}</h1>
<p>You are {{ data.age }} years old.</p>
<ul>{% for item in data.items %}<li>{{ item }}</li>{% endfor %}
</ul>
4.模板上下文处理器
上下文处理器是在渲染模板之前运行的函数,允许您自动将公共变量传递给所有模板。
例子:
@app.context_processor
def inject_user():return {'user': 'John Doe'}
这将注入user
到每个模板中,因此您可以{{ user }}
在任何模板中使用它而无需明确传递它。这个有点类似Node.js 的中间件,在app.j中定义好之后,继续往下路由,可以全局使用。
5.使用自定义错误页面处理错误
Flask 提供了一种定义自定义错误页面的方法(例如 404 或 500 错误)。你可以为这些错误渲染自定义模板。
例子:
- 包括:用于
{% include 'filename.html' %}
重用模板片段。 - 动态渲染
render_template_string
:有时,您可能需要从字符串而不是文件渲染模板。 - 异步渲染:对于需要异步处理的繁重操作,您可以使用 Flask 对后台作业或流式传输数据到客户端的支持。
6.高级模板技术
- 包括:用于
{% include 'filename.html' %}
重用模板片段。 - 动态渲染
render_template_string
:有时,您可能需要从字符串而不是文件渲染模板。 - 异步渲染:对于需要异步处理的繁重操作,您可以使用 Flask 对后台作业或流式传输数据到客户端的支持。
7.测试模板
Flask 有一个内置测试客户端,你可以用它来测试你的路由和模板。使用flask.testing.TestCase
,你可以断言模板的渲染是否正确。
from flask import Flask, render_template
import unittestapp = Flask(__name__)@app.route('/')
def home():return render_template('home.html', name="Test")class TemplateTest(unittest.TestCase):def setUp(self):app.config['TESTING'] = Trueself.client = app.test_client()def test_home_page(self):response = self.client.get('/')self.assertIn(b'Hello, Test', response.data)if __name__ == '__main__':unittest.main()
8.Flask扩展
Flask 有许多可以简化模板的扩展,例如:
- Flask-WTF:用于将表单集成到模板中。
- Flask-Login:用于管理模板中的用户会话。
除了上面这些,其他还有 循环、嵌套等方法,都很容易实现,这里就不一一赘述了。
通过学习这些高级特性,你可以更灵活地使用 Flask 的模板系统,编写更加干净、可维护和扩展的模板。你可以尝试将这些功能组合在一起,用于实现更复杂的页面和功能。如果有任何问题或不清楚的地方,欢迎随时留言一起来讨论!