Python Flask 编程 | 连载 07 - Jinja2 语法

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
简介: Python Flask 编程 | 连载 07 - Jinja2 语法

一、Jinja2 语法

模板标签

for循环表达式

在页面中展示列表或者字典数据时常常会用到 for循环,将列表中的每个数据迭代并进行展示,模板中 for 循环 的写法如下:

{% for key, value in data.items() %}
    {{ key }}: {{ value }}
{% else %}
    {# 展示字典为空的提示
{% endfor %}
复制代码

修改 tag 视图函数,增加一个含有字典数据的变量。

@app.route('/tag')
def tag():
    name='stark'
    heros = [
        {'name': 'stark', 'address': 'New York'},
        {'name': 'thor', 'address': 'Asgard'},
        {'name': 'strange', 'address': 'New York'},
        {'name': 'peter', 'address': 'New York Queens'},
    ]
    return render_template('tag.html', name=name, heros=heros)
复制代码

在 tag.html 的 body 标签中增加 for循环 表达式

<h2>for循环的使用</h2>
<ul>
    {% for hero in heros %}
        <li>{{ hero.name }}: {{ hero.address }}</li>
    {% else %}
        <li>NO Data</li>
    {% endfor %}
</ul>
复制代码

在浏览器中访问 /tag

image.png

for 循环 中包含了一个 loop 对象,即指代这次迭代,loop 对象中包含了一些属性如下:

变量 说明
loop.index 当前循环迭代次数从1开始
loop.index0 当前循环迭代次数从0开始
loop.revindex 循环从1开始到结束需要迭代的次数
loop.revindex0 循环从0开始到结束需要迭代的次数
loop.first 如果是第一次迭代,该变量的值为True,否则为False
loop.last 如果是最后一次迭代,该变量的值为True,否则为False
loop.length 迭代序列的长度
loop.cycle 在一串序列间取值的辅助函数

修改 tag.html,在 head 标签中增加样式并在 body 标签中使用 loop 对象的属性。

<head>
    <style>
        .yankee {
            background: yellow;
        }
        .zulu {
            background: red;
        }
    </style>
</head>
<body>
    <h2>for循环中loop对象的使用</h2>
    {% for hero in heros %}
        <p class="{{ loop.cycle('yankee', 'zulu')}}">
            当前是第{{ loop.index }}个元素,索引为{{ loop.index0 }},共{{ loop.length }}个元素
            {% for key, value in hero.items() %}
                {{ key }}: {{ value }}
            {% endfor %}
        </p>
    {% endfor %}
<body>
复制代码

再次访问 /tag

image.png

在 Python 代码中进行 for 循环 时可以通过 continue 或者 break 关键字来跳过当前循环或者停止循环,但是在模板中是没有 continuebreak 关键字的。

在 tag.html 中增加的 for循环,增加条件判断,当符合条件时使用 break 关键字结束循环。

<h2>for循环中使用break关键字</h2>
{% for hero in heros %}
    <p class="{{ loop.cycle('yankee', 'zulu')}}">
        当前是第{{ loop.index }}个元素,索引为{{ loop.index0 }},共{{ loop.length }}个元素
        {% for key, value in hero.items() %}
            {# 符合条件时退出循环,这里的loop.index是循环字典时的loop,注意与上面循环列表的loop进行区分 #}
            {% if loop.index == 2 %}
                {% break %}
            {% endif %}
            {{ key }}: {{ value }}
        {% endfor %}
    </p>
{% endfor %}
复制代码

image.png

此时页面报错出现未知的标签 break,说明在模板中是无法通过 break 来结束循环的。应在模板中添加扩展之后才可以 breakcontinue关键字。

在 app.py 中添加使用扩展的代码

app = Flask(__name__)
# 为模板引擎添加扩展,支持break/continue关键字
app.jinja_env.add_extension('jinja2.ext.loopcontrols')
复制代码

再次访问 /tag

image.png

这一次添加扩展之后,模板中 break 关键字生效。

使用标签渲染出来的 html 代码多了很多空行,这是会占用带宽的,去除这些空行有利于提升性能。

image.png

% 前后添加 - 可以删除空行。

<h2>for循环中使用break关键字</h2>
{% for hero in heros -%}
    <p class="{{ loop.cycle('yankee', 'zulu')}}">
        当前是第{{ loop.index }}个元素,索引为{{ loop.index0 }},共{{ loop.length }}个元素
        {%- for key, value in hero.items() -%}
            {%- if loop.index == 2 -%}
                {% break %}
            {%- endif -%}
            {{ key }}: {{ value }}
        {%- endfor -%}
    </p>
{% endfor %}
复制代码

再次查看页面的源代码,空格已被删除。

image.png

模板中的赋值

在模板中可以通过 set 关键字在 {%%} 中定义一个变量并进行赋值操作,set 关键字常与 with 关键字搭配使用,通过 with 关键字定义代码块,使得 set 关键字定义的变量只能在 with 指定的代码块使用。

{% with %}
{% set classname='zulu' %}
{{ classname }}
{% endwith %}
复制代码

在 tag.html 中增加如下内容:

<h2>set,with关键字的使用</h2>
{% with %}
    {% set classname='zulu' %}
    {# 在with代码块中定义了classname的值为zulu,背景色会变成红色 #}
    <p class="{{ classname }}">使用classname变量的p标签</p>
{% endwith %}
{# 在with代码块外,classname没有值 #}
<p class="{{ classname }}">使用classname变量的p标签</p>
复制代码

在浏览器中访问 /tag

image.png

模板标签特殊字符的转义

模板中的 {{}}{%%} 来进行渲染操作,那么如何在模板中显示这些特殊字符呢?

第一种方式是将这些特殊字符普通字符串处理;

<h2>模板语法中的特殊字符显示</h2>
<p>第一种方式:{{ '{{ value }}, {% if value=="stark" %}' }}</p>
复制代码

image.png

当代码比较多的时候,这种方式处理起来就比较繁琐。

第二种方式是使用 raw 标签;

<h2>模板语法中的特殊字符显示</h2>
<p>第一种方式:{{ '{{ value }}, {% if value=="stark" %}' }}</p>
<p>第二种方式:
    {% raw %}
    {{ value }}, {% if value=="stark" %}
    {% endraw %}
</p>
复制代码

image.png

推荐使用第二种方式。

模板的全局函数

全局函数可以在模板中直接使用,常用的全局函数有:

  • range(),返回一系列连续增加的整数
  • dict(),创建字典
  • cycler(),用于 CSS 类名循环
  • joiner(),字符串拼接
  • url_for(),URL 解析,用于静态文件地址解析、链接跳转地址解析
  • get_flashed_message():会话消息,如登录成功或者更新成功的提示

在 templates 目录下新建一个 global_funcs.html,内容如下:

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8"/>
    <title>全局函数的使用</title>
</head>
<body>
    <h2>全局函数range()函数的使用</h2>
    {% for i in range(7) %}
        <p>{{ i }}</p>
    {% endfor %}
</body>
</html>
复制代码

在 app.py 中新建一个视图函数 global_funcs,返回 global_funcs.html 页面。

@app.route('/global_funcs')
def gloab_funcs():
    return render_template('global_funcs.html')
复制代码

image.png

模板中的 range 函数返回了从 0~6 之间的整数。

在 global_funcs.html 中新增 cycler 函数;

<head>
    <meta charset="UTF-8"/>
    <title>全局函数的使用</title>
    <style>
        .zulu {
            color: red;
        }
        .yankee {
            color: aqua;
        }
    </style>
</head>
<body>
    <h2>全局函数cycle()函数的使用</h2>
    {% set class_name = cycler('zulu', 'yankee') %}
    {% for i in range(7) %}
        <p class={{ class_name.next() }}>{{ i }}</p>
    {% endfor %}
</body>
复制代码

image.png

cycle 函数轮询 zulu 和 yankee 两个 CSS 属性,给 range 函数生成的 7 个 p 标签填了这两个 CSS 属性。

在模板中使用 url_for 进行解析静态文件地址,新建一个 static 文件夹,并在该文件夹下新增一个 CSS 文件。

body {
    background-color: cadetblue;
}
h2 {
    color: white;
}
复制代码

在 global_funcs.html 页面中的 head 增加引用。

<head>
    <!--其余代码不变-->
    <link rel="stylesheet" href="{{ url_for('static', filename='style.css') }}">
</head>
复制代码

image.png

样式发生改变,url_for 函数可以正确解析出 CSS 的路径。如果没有生效,清除缓存即可。

url_for 还可以根据 url name 解析出 url 映射地址。

使用 url_for 首先要给 url 设置一个 url name,在 app.py 中增加代码:

@app.route('/index')
def show_html():
    # 其余代码保持不变
# 给url设置一个name
app.add_url_rule('/index', 'index')
复制代码

修改 global_funcs.html 页面内容如下:

<body>
    <h2>全局函数url_for的使用</h2>
    <a href="{{ url_for('index') }}">INDEX</a>
</body>
复制代码

访问 /global_funcs

image.png

点击INDEX,即可跳转到 /index 页面。

image.png


相关文章
|
3天前
|
机器学习/深度学习 人工智能 TensorFlow
人工智能浪潮下的自我修养:从Python编程入门到深度学习实践
【10月更文挑战第39天】本文旨在为初学者提供一条清晰的道路,从Python基础语法的掌握到深度学习领域的探索。我们将通过简明扼要的语言和实际代码示例,引导读者逐步构建起对人工智能技术的理解和应用能力。文章不仅涵盖Python编程的基础,还将深入探讨深度学习的核心概念、工具和实战技巧,帮助读者在AI的浪潮中找到自己的位置。
|
3天前
|
机器学习/深度学习 数据挖掘 Python
Python编程入门——从零开始构建你的第一个程序
【10月更文挑战第39天】本文将带你走进Python的世界,通过简单易懂的语言和实际的代码示例,让你快速掌握Python的基础语法。无论你是编程新手还是想学习新语言的老手,这篇文章都能为你提供有价值的信息。我们将从变量、数据类型、控制结构等基本概念入手,逐步过渡到函数、模块等高级特性,最后通过一个综合示例来巩固所学知识。让我们一起开启Python编程之旅吧!
|
3天前
|
存储 Python
Python编程入门:打造你的第一个程序
【10月更文挑战第39天】在数字时代的浪潮中,掌握编程技能如同掌握了一门新时代的语言。本文将引导你步入Python编程的奇妙世界,从零基础出发,一步步构建你的第一个程序。我们将探索编程的基本概念,通过简单示例理解变量、数据类型和控制结构,最终实现一个简单的猜数字游戏。这不仅是一段代码的旅程,更是逻辑思维和问题解决能力的锻炼之旅。准备好了吗?让我们开始吧!
|
5天前
|
设计模式 算法 搜索推荐
Python编程中的设计模式:优雅解决复杂问题的钥匙####
本文将探讨Python编程中几种核心设计模式的应用实例与优势,不涉及具体代码示例,而是聚焦于每种模式背后的设计理念、适用场景及其如何促进代码的可维护性和扩展性。通过理解这些设计模式,开发者可以更加高效地构建软件系统,实现代码复用,提升项目质量。 ####
|
4天前
|
机器学习/深度学习 存储 算法
探索Python编程:从基础到高级应用
【10月更文挑战第38天】本文旨在引导读者从Python的基础知识出发,逐渐深入到高级编程概念。通过简明的语言和实际代码示例,我们将一起探索这门语言的魅力和潜力,理解它如何帮助解决现实问题,并启发我们思考编程在现代社会中的作用和意义。
|
5天前
|
机器学习/深度学习 数据挖掘 开发者
Python编程入门:理解基础语法与编写第一个程序
【10月更文挑战第37天】本文旨在为初学者提供Python编程的初步了解,通过简明的语言和直观的例子,引导读者掌握Python的基础语法,并完成一个简单的程序。我们将从变量、数据类型到控制结构,逐步展开讲解,确保即使是编程新手也能轻松跟上。文章末尾附有完整代码示例,供读者参考和实践。
|
5天前
|
人工智能 数据挖掘 程序员
Python编程入门:从零到英雄
【10月更文挑战第37天】本文将引导你走进Python编程的世界,无论你是初学者还是有一定基础的开发者,都能从中受益。我们将从最基础的语法开始讲解,逐步深入到更复杂的主题,如数据结构、面向对象编程和网络编程等。通过本文的学习,你将能够编写出自己的Python程序,实现各种功能。让我们一起踏上Python编程之旅吧!
|
6天前
|
数据采集 机器学习/深度学习 人工智能
Python编程入门:从基础到实战
【10月更文挑战第36天】本文将带你走进Python的世界,从基础语法出发,逐步深入到实际项目应用。我们将一起探索Python的简洁与强大,通过实例学习如何运用Python解决问题。无论你是编程新手还是希望扩展技能的老手,这篇文章都将为你提供有价值的指导和灵感。让我们一起开启Python编程之旅,用代码书写想法,创造可能。
|
8天前
|
Python
不容错过!Python中图的精妙表示与高效遍历策略,提升你的编程艺术感
本文介绍了Python中图的表示方法及遍历策略。图可通过邻接表或邻接矩阵表示,前者节省空间适合稀疏图,后者便于检查连接但占用更多空间。文章详细展示了邻接表和邻接矩阵的实现,并讲解了深度优先搜索(DFS)和广度优先搜索(BFS)的遍历方法,帮助读者掌握图的基本操作和应用技巧。
25 4
|
10天前
|
存储 人工智能 数据挖掘
从零起步,揭秘Python编程如何带你从新手村迈向高手殿堂
【10月更文挑战第32天】Python,诞生于1991年的高级编程语言,以其简洁明了的语法成为众多程序员的入门首选。从基础的变量类型、控制流到列表、字典等数据结构,再到函数定义与调用及面向对象编程,Python提供了丰富的功能和强大的库支持,适用于Web开发、数据分析、人工智能等多个领域。学习Python不仅是掌握一门语言,更是加入一个充满活力的技术社区,开启探索未知世界的旅程。
20 6