0

0

Flask与Jinja2:正确传递URL参数以避免404错误

碧海醫心

碧海醫心

发布时间:2025-09-28 10:21:07

|

1428人浏览过

|

来源于php中文网

原创

Flask与Jinja2:正确传递URL参数以避免404错误

本文旨在解决Flask应用中因Jinja2模板URL参数传递不当导致的"404 Not Found"错误。通过分析一个密码更新功能的典型场景,我们将深入探讨Jinja2变量渲染的正确语法,并提供修复方案,确保表单数据能够正确提交到指定路由,从而避免常见的路由匹配失败问题。

在开发web应用时,flask框架与jinja2模板引擎的结合为我们提供了强大的功能。然而,在处理动态url和表单提交时,一个常见的陷阱是jinja2变量渲染语法的误用,这可能导致“404 not found”错误。本教程将深入探讨这一问题,并提供清晰的解决方案。

理解“404 Not Found”错误的根源

当用户尝试通过表单提交数据,但服务器返回“404 Not Found”错误时,通常意味着客户端请求的URL与服务器上定义的任何路由都不匹配。在Flask应用中,这尤其可能发生在动态URL参数的传递上。

考虑以下HTML表单代码片段:

{{ form.password(class="form-control", value=userPass.password) }} {{ form.submit}}

以及对应的Flask路由:

@app.route('/update/', methods=['GET', 'POST'])
@login_required
def update(id):
    # ... 省略部分代码 ...
    pass

问题在于HTML表单中的action="/update/{id}"。尽管Flask路由定义了作为动态参数,但Jinja2模板引擎在渲染HTML时,并不会将{id}解析为一个变量。它会将其视为字面量字符串。因此,当浏览器提交表单时,它会向/update/{id}这个确切的URL发送请求,而不是将{id}替换为实际的用户ID。由于Flask应用中没有名为/update/{id}的路由,服务器自然会返回“404 Not Found”错误。

Jinja2变量渲染机制

Jinja2是一个功能强大的模板引擎,它使用特定的语法来识别和渲染模板中的变量、控制结构等。对于变量,Jinja2的规定是使用双大括号{{ variable_name }}。

例如,如果你想在模板中显示一个名为id的变量的值,你应该这样写:

用户ID是:{{ id }}

只有这样,Jinja2才会识别id为一个变量,并将其替换为后端传递过来的实际值。

修复方案:正确传递URL参数

要解决上述“404 Not Found”问题,只需将HTML表单中的action属性修改为正确的Jinja2变量渲染语法:

Digram
Digram

让Figma更好用的AI神器

下载

错误的写法:

正确的写法:

通过将{id}更改为{{id}},Jinja2模板引擎在渲染HTML时,会将{{id}}替换为update函数中传入的id变量的实际值。例如,如果id的值是123,那么渲染后的HTML将是:

此时,浏览器将向/update/123发送POST请求,这与Flask路由@app.route('/update/')完美匹配,从而避免了“404 Not Found”错误,并允许后端正确处理数据。

完整的代码示例(修正后)

修正后的HTML模板 (update.html):



    
    
    
    Update Page


Update Password

{# 注意这里将 {id} 改为 {{id}} #} {{ form.csrf_token }} {# 推荐添加CSRF保护 #} {{ form.password(class="form-control", value=userPass.password) }} {{ form.submit}}

Flask后端代码 (app.py):

from flask import Flask, render_template, request, redirect, url_for, flash
from flask_sqlalchemy import SQLAlchemy
from flask_login import LoginManager, login_required, UserMixin # 假设已配置Flask-Login
from flask_wtf import FlaskForm
from wtforms import PasswordField, SubmitField
from wtforms.validators import DataRequired

# 假设的Flask应用和数据库配置
app = Flask(__name__)
app.config['SECRET_KEY'] = 'your_secret_key' # 生产环境请使用强密钥
app.config['SQLALCHEMY_DATABASE_URI'] = 'sqlite:///users.db'
db = SQLAlchemy(app)
login_manager = LoginManager(app)
login_manager.login_view = 'login'

# 假设的用户模型
class User(db.Model, UserMixin):
    id = db.Column(db.Integer, primary_key=True)
    username = db.Column(db.String(80), unique=True, nullable=False)
    password = db.Column(db.String(120), nullable=False)

    def get_id(self):
        return str(self.id)

@login_manager.user_loader
def load_user(user_id):
    return User.query.get(int(user_id))

# 假设的密码更新表单
class UpdatePasswordForm(FlaskForm):
    password = PasswordField('New Password', validators=[DataRequired()])
    submit = SubmitField('Update Password')

# 假设的登录路由
@app.route('/login')
def login():
    return "Login Page (Placeholder)" # 实际应渲染登录模板

@app.route('/update/', methods=['GET', 'POST'])
@login_required
def update(id):
    form = UpdatePasswordForm()
    userPass = User.query.get_or_404(id) # 使用get_or_404处理用户不存在的情况

    if request.method == "POST":
        if form.validate_on_submit(): # 验证表单数据
            # 注意:在实际应用中,密码应进行哈希处理,例如使用 werkzeug.security.generate_password_hash
            userPass.password = form.password.data # 从表单获取数据
            try:
                db.session.commit()
                flash("User Updated Successfully!")
                return redirect(url_for('login')) # 成功后重定向到登录页
            except Exception as e: # 捕获更具体的异常,并回滚
                db.session.rollback()
                flash(f"Error! There was a problem changing your password: {e}")
                # 错误时,重新渲染页面并显示错误信息
                return render_template("update.html",
                                       form=form,
                                       userPass=userPass,
                                       id=id)
        else:
            # 表单验证失败,重新渲染页面并显示验证错误
            for field, errors in form.errors.items():
                for error in errors:
                    flash(f"Error in {field}: {error}")
            return render_template("update.html",
                                   form=form,
                                   userPass=userPass,
                                   id=id)
    else:
        # GET请求,初始化表单(密码字段通常不预填充旧密码)
        return render_template("update.html",
                               form=form,
                               userPass=userPass,
                               id=id)

if __name__ == '__main__':
    with app.app_context():
        db.create_all()
        # 示例用户(如果需要)
        # if not User.query.filter_by(username='testuser').first():
        #     new_user = User(username='testuser', password='oldpassword') # 实际应哈希
        #     db.session.add(new_user)
        #     db.session.commit()
    app.run(debug=True)

注意事项与最佳实践:

  1. Jinja2变量语法: 始终记住使用{{ variable }}来渲染变量。对于控制结构(如if、for),则使用{% control_statement %}。
  2. CSRF保护: 在表单中添加{{ form.csrf_token }}以启用Flask-WTF提供的跨站请求伪造(CSRF)保护,增强应用安全性。
  3. 密码哈希: 在实际应用中,绝不应将用户密码以明文形式存储在数据库中。务必使用werkzeug.security.generate_password_hash和check_password_hash等工具对密码进行哈希处理。
  4. 错误处理: 在try...except块中,捕获更具体的异常,并在发生数据库错误时执行db.session.rollback(),以撤销未提交的更改。
  5. 重定向逻辑: 确保redirect(url_for(...))语句能够被执行,它通常作为return语句的一部分。在原始代码中,redirect语句位于return render_template之后,导致其永远不会被执行。
  6. 表单验证: 充分利用Flask-WTF提供的表单验证功能,通过form.validate_on_submit()在后端对用户输入进行验证,提高数据的完整性和安全性。

总结

正确理解和运用Jinja2的变量渲染语法是构建健壮Flask应用的关键。通过将form action="/update/{id}"修正为form action="/update/{{id}}",我们解决了因URL参数传递不当导致的“404 Not Found”错误。同时,本教程也强调了在处理用户数据和表单提交时,应遵循的安全最佳实践,如密码哈希、CSRF保护和完善的错误处理机制。掌握这些核心概念,将有助于您开发出更加稳定、安全的Web应用。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
Python Flask框架
Python Flask框架

本专题专注于 Python 轻量级 Web 框架 Flask 的学习与实战,内容涵盖路由与视图、模板渲染、表单处理、数据库集成、用户认证以及RESTful API 开发。通过博客系统、任务管理工具与微服务接口等项目实战,帮助学员掌握 Flask 在快速构建小型到中型 Web 应用中的核心技能。

86

2025.08.25

Python Flask Web框架与API开发
Python Flask Web框架与API开发

本专题系统介绍 Python Flask Web框架的基础与进阶应用,包括Flask路由、请求与响应、模板渲染、表单处理、安全性加固、数据库集成(SQLAlchemy)、以及使用Flask构建 RESTful API 服务。通过多个实战项目,帮助学习者掌握使用 Flask 开发高效、可扩展的 Web 应用与 API。

72

2025.12.15

if什么意思
if什么意思

if的意思是“如果”的条件。它是一个用于引导条件语句的关键词,用于根据特定条件的真假情况来执行不同的代码块。本专题提供if什么意思的相关文章,供大家免费阅读。

775

2023.08.22

session失效的原因
session失效的原因

session失效的原因有会话超时、会话数量限制、会话完整性检查、服务器重启、浏览器或设备问题等等。详细介绍:1、会话超时:服务器为Session设置了一个默认的超时时间,当用户在一段时间内没有与服务器交互时,Session将自动失效;2、会话数量限制:服务器为每个用户的Session数量设置了一个限制,当用户创建的Session数量超过这个限制时,最新的会覆盖最早的等等。

315

2023.10.17

session失效解决方法
session失效解决方法

session失效通常是由于 session 的生存时间过期或者服务器关闭导致的。其解决办法:1、延长session的生存时间;2、使用持久化存储;3、使用cookie;4、异步更新session;5、使用会话管理中间件。

748

2023.10.18

cookie与session的区别
cookie与session的区别

本专题整合了cookie与session的区别和使用方法等相关内容,阅读专题下面的文章了解更详细的内容。

91

2025.08.19

js 字符串转数组
js 字符串转数组

js字符串转数组的方法:1、使用“split()”方法;2、使用“Array.from()”方法;3、使用for循环遍历;4、使用“Array.split()”方法。本专题为大家提供js字符串转数组的相关的文章、下载、课程内容,供大家免费下载体验。

298

2023.08.03

js截取字符串的方法
js截取字符串的方法

js截取字符串的方法有substring()方法、substr()方法、slice()方法、split()方法和slice()方法。本专题为大家提供字符串相关的文章、下载、课程内容,供大家免费下载体验。

212

2023.09.04

Python 自然语言处理(NLP)基础与实战
Python 自然语言处理(NLP)基础与实战

本专题系统讲解 Python 在自然语言处理(NLP)领域的基础方法与实战应用,涵盖文本预处理(分词、去停用词)、词性标注、命名实体识别、关键词提取、情感分析,以及常用 NLP 库(NLTK、spaCy)的核心用法。通过真实文本案例,帮助学习者掌握 使用 Python 进行文本分析与语言数据处理的完整流程,适用于内容分析、舆情监测与智能文本应用场景。

10

2026.01.27

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Bootstrap 5教程
Bootstrap 5教程

共46课时 | 3万人学习

AngularJS教程
AngularJS教程

共24课时 | 3万人学习

CSS教程
CSS教程

共754课时 | 24.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号