Python之Falsk开发框架(第四篇)- Flask 知识总结与完整博客系统实战

经过前三篇的系统学习,我们已经掌握了 Flask 从入门到进阶的完整知识体系。本篇将对 Flask 的核心知识进行总结,并在此基础上提供一个完整的、可直接运行的简易博客系统,将所学内容融会贯通,帮助你巩固知识并快速应用于实际项目。


一、Flask 知识体系回顾

1. 核心概念

  • 应用(Application)Flask 类的实例,是整个应用的入口。
  • 路由(Route) :使用 @app.route 装饰器将 URL 映射到视图函数,支持动态 URL 和多种 HTTP 方法。
  • 视图函数(View Function) :处理请求并返回响应,支持返回字符串、字典、元组、Response 对象等。
  • 请求(Request)request 对象封装了客户端发送的请求信息,如表单数据、查询参数、JSON 数据、请求头等。
  • 响应(Response) :视图函数的返回值会被 Flask 自动转换为 Response 对象,可以手动构建更复杂的响应。
  • 会话(Session):基于签名 cookie 的会话管理,用于跨请求保存用户数据。

2. 模板与静态文件

  • Jinja2 模板引擎:支持变量、控制结构、过滤器、宏、模板继承。
  • 静态文件 :存放在 static 目录,通过 url_for('static', filename='...') 生成 URL。

3. 表单处理

  • Flask-WTF 扩展集成 WTForms,提供 CSRF 保护、表单验证、字段渲染等功能。
  • 表单类:定义字段和验证器,在视图函数中实例化并验证。

4. 数据库操作

  • Flask-SQLAlchemy 提供 ORM 能力,简化数据库操作。
  • 模型定义 :继承 db.Model,使用 db.Column 定义字段,支持关系映射(一对多、多对多)。
  • 迁移管理 :通过 Flask-Migrate 自动化生成迁移脚本,管理数据库版本。

5. 蓝图(Blueprint)

  • 模块化 :将相关路由、视图、模板等组织成蓝图,通过 app.register_blueprint 注册到应用。
  • URL 前缀 :在蓝图创建时指定 url_prefix,实现路由分组。

6. 请求钩子与中间件

  • before_requestafter_requestteardown_request:在请求生命周期各阶段执行通用逻辑。
  • WSGI 中间件:将 Flask 应用包装在 WSGI 中间件中,扩展功能(如性能分析、代理处理)。

7. 常用扩展

扩展 用途
Flask-Login 用户认证,管理登录会话
Flask-Mail 发送电子邮件
Flask-Caching 缓存视图或函数结果
Flask-RQ2 基于 Redis 的后台任务队列
Flask-RESTx 构建 RESTful API
Flask-SocketIO WebSocket 实时通信

8. 生产部署

  • WSGI 服务器:Gunicorn、uWSGI 等。
  • 反向代理:Nginx 处理静态文件、负载均衡、HTTPS 等。
  • 进程管理:Supervisor、systemd 等保证应用持续运行。
  • 容器化:Docker 打包应用与环境,便于部署和扩展。
  • 云平台:Heroku、PythonAnywhere、AWS 等。

二、完整简易博客系统

下面我们将前面的知识点整合成一个功能完备的博客系统。该系统具有以下特性:

  • 用户注册、登录、登出(使用 Flask-Login)
  • 注册时发送欢迎邮件(使用 Flask-Mail)
  • 文章创建、编辑、删除(仅作者本人)
  • 文章列表分页显示
  • 首页缓存(使用 Flask-Caching)
  • 使用蓝图组织路由
  • 使用 SQLite 数据库(可轻松切换至 MySQL/PostgreSQL)
  • 生产配置与环境变量管理

1. 项目结构

复制代码
blog/
├── app/
│   ├── __init__.py
│   ├── models.py
│   ├── forms.py
│   ├── auth/
│   │   ├── __init__.py
│   │   └── routes.py
│   ├── main/
│   │   ├── __init__.py
│   │   └── routes.py
│   ├── templates/
│   │   ├── base.html
│   │   ├── auth/
│   │   │   ├── login.html
│   │   │   └── register.html
│   │   └── main/
│   │       ├── index.html
│   │       ├── post_detail.html
│   │       ├── create_post.html
│   │       └── edit_post.html
│   └── static/
│       └── style.css
├── config.py
├── run.py
├── requirements.txt
└── .env

2. 环境准备

创建虚拟环境并安装依赖:

bash 复制代码
python -m venv venv
source venv/bin/activate   # Windows: venv\Scripts\activate
pip install flask flask-sqlalchemy flask-migrate flask-wtf flask-login flask-mail flask-caching python-dotenv

requirements.txt 内容:

复制代码
flask==2.3.3
flask-sqlalchemy==3.1.1
flask-migrate==4.0.5
flask-wtf==1.1.1
flask-login==0.6.2
flask-mail==0.9.1
flask-caching==2.1.0
python-dotenv==1.0.0

3. 配置文件 config.py

python 复制代码
import os
from dotenv import load_dotenv

load_dotenv()

class Config:
    SECRET_KEY = os.environ.get('SECRET_KEY') or 'hard-to-guess-string'
    SQLALCHEMY_DATABASE_URI = os.environ.get('DATABASE_URL') or 'sqlite:///blog.db'
    SQLALCHEMY_TRACK_MODIFICATIONS = False
    
    # Flask-Mail 配置
    MAIL_SERVER = os.environ.get('MAIL_SERVER', 'smtp.qq.com')
    MAIL_PORT = int(os.environ.get('MAIL_PORT', 587))
    MAIL_USE_TLS = os.environ.get('MAIL_USE_TLS', 'true').lower() in ['true', 'on', '1']
    MAIL_USERNAME = os.environ.get('MAIL_USERNAME')
    MAIL_PASSWORD = os.environ.get('MAIL_PASSWORD')
    
    # Flask-Caching 配置
    CACHE_TYPE = os.environ.get('CACHE_TYPE', 'SimpleCache')

4. 应用工厂 app/__init__.py

python 复制代码
from flask import Flask
from flask_sqlalchemy import SQLAlchemy
from flask_migrate import Migrate
from flask_login import LoginManager
from flask_mail import Mail
from flask_caching import Cache
from config import Config

db = SQLAlchemy()
migrate = Migrate()
login_manager = LoginManager()
mail = Mail()
cache = Cache()

def create_app(config_class=Config):
    app = Flask(__name__)
    app.config.from_object(config_class)
    
    db.init_app(app)
    migrate.init_app(app, db)
    login_manager.init_app(app)
    mail.init_app(app)
    cache.init_app(app)
    
    login_manager.login_view = 'auth.login'
    login_manager.login_message = '请先登录'
    
    from app.auth import auth_bp
    from app.main import main_bp
    app.register_blueprint(auth_bp)
    app.register_blueprint(main_bp)
    
    return app

5. 数据库模型 app/models.py

python 复制代码
from app import db
from flask_login import UserMixin
from werkzeug.security import generate_password_hash, check_password_hash
from datetime import datetime

class User(UserMixin, db.Model):
    id = db.Column(db.Integer, primary_key=True)
    username = db.Column(db.String(64), unique=True, nullable=False)
    email = db.Column(db.String(120), unique=True, nullable=False)
    password_hash = db.Column(db.String(128))
    posts = db.relationship('Post', backref='author', lazy='dynamic')
    
    def set_password(self, password):
        self.password_hash = generate_password_hash(password)
    
    def check_password(self, password):
        return check_password_hash(self.password_hash, password)
    
    def __repr__(self):
        return f'<User {self.username}>'

class Post(db.Model):
    id = db.Column(db.Integer, primary_key=True)
    title = db.Column(db.String(100), nullable=False)
    body = db.Column(db.Text, nullable=False)
    created_at = db.Column(db.DateTime, default=datetime.utcnow)
    user_id = db.Column(db.Integer, db.ForeignKey('user.id'))
    
    def __repr__(self):
        return f'<Post {self.title}>'

6. 表单类 app/forms.py

python 复制代码
from flask_wtf import FlaskForm
from wtforms import StringField, PasswordField, SubmitField, TextAreaField
from wtforms.validators import DataRequired, Email, Length, EqualTo

class RegistrationForm(FlaskForm):
    username = StringField('用户名', validators=[DataRequired(), Length(min=2, max=64)])
    email = StringField('邮箱', validators=[DataRequired(), Email()])
    password = PasswordField('密码', validators=[DataRequired(), Length(min=6)])
    confirm = PasswordField('确认密码', validators=[DataRequired(), EqualTo('password')])
    submit = SubmitField('注册')

class LoginForm(FlaskForm):
    email = StringField('邮箱', validators=[DataRequired(), Email()])
    password = PasswordField('密码', validators=[DataRequired()])
    submit = SubmitField('登录')

class PostForm(FlaskForm):
    title = StringField('标题', validators=[DataRequired()])
    body = TextAreaField('内容', validators=[DataRequired()])
    submit = SubmitField('发布')

7. 认证蓝图 app/auth/__init__.py

python 复制代码
from flask import Blueprint

auth_bp = Blueprint('auth', __name__, url_prefix='/auth')

from app.auth import routes

8. 认证路由 app/auth/routes.py

python 复制代码
from flask import render_template, redirect, url_for, flash, request
from flask_login import login_user, logout_user, current_user
from app import db, mail
from app.auth import auth_bp
from app.forms import RegistrationForm, LoginForm
from app.models import User
from flask_mail import Message

@auth_bp.route('/register', methods=['GET', 'POST'])
def register():
    if current_user.is_authenticated:
        return redirect(url_for('main.index'))
    form = RegistrationForm()
    if form.validate_on_submit():
        # 检查用户是否存在
        user = User.query.filter_by(username=form.username.data).first()
        if user:
            flash('用户名已存在', 'danger')
            return redirect(url_for('auth.register'))
        user = User.query.filter_by(email=form.email.data).first()
        if user:
            flash('邮箱已被注册', 'danger')
            return redirect(url_for('auth.register'))
        
        user = User(username=form.username.data, email=form.email.data)
        user.set_password(form.password.data)
        db.session.add(user)
        db.session.commit()
        
        # 发送欢迎邮件
        try:
            msg = Message('欢迎注册博客', recipients=[user.email])
            msg.body = f'你好 {user.username},感谢注册我们的博客系统!'
            mail.send(msg)
        except Exception as e:
            app.logger.error(f'邮件发送失败: {e}')
        
        flash('注册成功,请登录', 'success')
        return redirect(url_for('auth.login'))
    return render_template('auth/register.html', form=form)

@auth_bp.route('/login', methods=['GET', 'POST'])
def login():
    if current_user.is_authenticated:
        return redirect(url_for('main.index'))
    form = LoginForm()
    if form.validate_on_submit():
        user = User.query.filter_by(email=form.email.data).first()
        if user and user.check_password(form.password.data):
            login_user(user)
            next_page = request.args.get('next')
            return redirect(next_page) if next_page else redirect(url_for('main.index'))
        else:
            flash('邮箱或密码错误', 'danger')
    return render_template('auth/login.html', form=form)

@auth_bp.route('/logout')
def logout():
    logout_user()
    return redirect(url_for('main.index'))

9. 主蓝图 app/main/__init__.py

python 复制代码
from flask import Blueprint

main_bp = Blueprint('main', __name__)

from app.main import routes

10. 主路由 app/main/routes.py

python 复制代码
from flask import render_template, redirect, url_for, flash, abort, request
from flask_login import login_required, current_user
from app import db, cache
from app.main import main_bp
from app.forms import PostForm
from app.models import Post

@main_bp.route('/')
@cache.cached(timeout=60)
def index():
    page = request.args.get('page', 1, type=int)
    posts = Post.query.order_by(Post.created_at.desc()).paginate(page=page, per_page=10)
    return render_template('main/index.html', posts=posts)

@main_bp.route('/post/<int:id>')
def post_detail(id):
    post = Post.query.get_or_404(id)
    return render_template('main/post_detail.html', post=post)

@main_bp.route('/create', methods=['GET', 'POST'])
@login_required
def create_post():
    form = PostForm()
    if form.validate_on_submit():
        post = Post(title=form.title.data, body=form.body.data, author=current_user)
        db.session.add(post)
        db.session.commit()
        flash('文章发布成功', 'success')
        return redirect(url_for('main.index'))
    return render_template('main/create_post.html', form=form)

@main_bp.route('/edit/<int:id>', methods=['GET', 'POST'])
@login_required
def edit_post(id):
    post = Post.query.get_or_404(id)
    if post.author != current_user:
        abort(403)
    form = PostForm()
    if form.validate_on_submit():
        post.title = form.title.data
        post.body = form.body.data
        db.session.commit()
        flash('文章已更新', 'success')
        return redirect(url_for('main.post_detail', id=post.id))
    elif request.method == 'GET':
        form.title.data = post.title
        form.body.data = post.body
    return render_template('main/edit_post.html', form=form, post=post)

@main_bp.route('/delete/<int:id>', methods=['POST'])
@login_required
def delete_post(id):
    post = Post.query.get_or_404(id)
    if post.author != current_user:
        abort(403)
    db.session.delete(post)
    db.session.commit()
    flash('文章已删除', 'success')
    return redirect(url_for('main.index'))

11. 用户加载回调(在 app/__init__.py 中补充)

python 复制代码
from app.models import User

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

将这段代码放在 create_app 函数中,login_manager.init_app(app) 之后。

12. 模板文件

templates/base.html
html 复制代码
<!DOCTYPE html>
<html lang="zh">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>{% block title %}博客系统{% endblock %}</title>
    <link rel="stylesheet" href="{{ url_for('static', filename='style.css') }}">
</head>
<body>
    <nav>
        <div class="container">
            <a href="{{ url_for('main.index') }}" class="brand">我的博客</a>
            <ul class="nav-links">
                {% if current_user.is_authenticated %}
                    <li>你好,{{ current_user.username }}</li>
                    <li><a href="{{ url_for('main.create_post') }}">写文章</a></li>
                    <li><a href="{{ url_for('auth.logout') }}">登出</a></li>
                {% else %}
                    <li><a href="{{ url_for('auth.login') }}">登录</a></li>
                    <li><a href="{{ url_for('auth.register') }}">注册</a></li>
                {% endif %}
            </ul>
        </div>
    </nav>
    <main class="container">
        {% with messages = get_flashed_messages(with_categories=true) %}
            {% if messages %}
                {% for category, message in messages %}
                    <div class="alert alert-{{ category }}">{{ message }}</div>
                {% endfor %}
            {% endif %}
        {% endwith %}
        {% block content %}{% endblock %}
    </main>
    <footer>
        <div class="container">
            <p>&copy; 2023 我的博客</p>
        </div>
    </footer>
</body>
</html>
templates/auth/register.html
html 复制代码
{% extends "base.html" %}

{% block title %}注册{% endblock %}

{% block content %}
<h2>注册</h2>
<form method="POST">
    {{ form.hidden_tag() }}
    <div class="form-group">
        {{ form.username.label }}<br>
        {{ form.username(size=32) }}<br>
        {% for error in form.username.errors %}
            <span class="error">{{ error }}</span>
        {% endfor %}
    </div>
    <div class="form-group">
        {{ form.email.label }}<br>
        {{ form.email(size=32) }}<br>
        {% for error in form.email.errors %}
            <span class="error">{{ error }}</span>
        {% endfor %}
    </div>
    <div class="form-group">
        {{ form.password.label }}<br>
        {{ form.password(size=32) }}<br>
        {% for error in form.password.errors %}
            <span class="error">{{ error }}</span>
        {% endfor %}
    </div>
    <div class="form-group">
        {{ form.confirm.label }}<br>
        {{ form.confirm(size=32) }}<br>
        {% for error in form.confirm.errors %}
            <span class="error">{{ error }}</span>
        {% endfor %}
    </div>
    <div class="form-group">
        {{ form.submit() }}
    </div>
</form>
{% endblock %}
templates/auth/login.html
html 复制代码
{% extends "base.html" %}

{% block title %}登录{% endblock %}

{% block content %}
<h2>登录</h2>
<form method="POST">
    {{ form.hidden_tag() }}
    <div class="form-group">
        {{ form.email.label }}<br>
        {{ form.email(size=32) }}<br>
        {% for error in form.email.errors %}
            <span class="error">{{ error }}</span>
        {% endfor %}
    </div>
    <div class="form-group">
        {{ form.password.label }}<br>
        {{ form.password(size=32) }}<br>
        {% for error in form.password.errors %}
            <span class="error">{{ error }}</span>
        {% endfor %}
    </div>
    <div class="form-group">
        {{ form.submit() }}
    </div>
</form>
{% endblock %}
templates/main/index.html
html 复制代码
{% extends "base.html" %}

{% block title %}首页{% endblock %}

{% block content %}
<h1>最新文章</h1>
{% for post in posts.items %}
    <article class="post">
        <h2><a href="{{ url_for('main.post_detail', id=post.id) }}">{{ post.title }}</a></h2>
        <div class="post-meta">
            作者:{{ post.author.username }} | 发布于:{{ post.created_at.strftime('%Y-%m-%d') }}
        </div>
        <div class="post-summary">
            {{ post.body[:200] }}{% if post.body|length > 200 %}...{% endif %}
        </div>
        {% if current_user == post.author %}
            <div class="post-actions">
                <a href="{{ url_for('main.edit_post', id=post.id) }}">编辑</a>
                <form action="{{ url_for('main.delete_post', id=post.id) }}" method="POST" style="display:inline;">
                    <button type="submit" onclick="return confirm('确定删除吗?')">删除</button>
                </form>
            </div>
        {% endif %}
    </article>
{% else %}
    <p>暂无文章</p>
{% endfor %}

<div class="pagination">
    {% if posts.has_prev %}
        <a href="{{ url_for('main.index', page=posts.prev_num) }}">上一页</a>
    {% endif %}
    <span>第 {{ posts.page }} 页 / 共 {{ posts.pages }} 页</span>
    {% if posts.has_next %}
        <a href="{{ url_for('main.index', page=posts.next_num) }}">下一页</a>
    {% endif %}
</div>
{% endblock %}
templates/main/post_detail.html
html 复制代码
{% extends "base.html" %}

{% block title %}{{ post.title }}{% endblock %}

{% block content %}
<article class="post-detail">
    <h1>{{ post.title }}</h1>
    <div class="post-meta">
        作者:{{ post.author.username }} | 发布于:{{ post.created_at.strftime('%Y-%m-%d %H:%M') }}
    </div>
    <div class="post-body">
        {{ post.body|safe }}
    </div>
    {% if current_user == post.author %}
        <div class="post-actions">
            <a href="{{ url_for('main.edit_post', id=post.id) }}">编辑</a>
            <form action="{{ url_for('main.delete_post', id=post.id) }}" method="POST" style="display:inline;">
                <button type="submit" onclick="return confirm('确定删除吗?')">删除</button>
            </form>
        </div>
    {% endif %}
</article>
{% endblock %}
templates/main/create_post.html
html 复制代码
{% extends "base.html" %}

{% block title %}写文章{% endblock %}

{% block content %}
<h2>发布新文章</h2>
<form method="POST">
    {{ form.hidden_tag() }}
    <div class="form-group">
        {{ form.title.label }}<br>
        {{ form.title(size=64) }}<br>
        {% for error in form.title.errors %}
            <span class="error">{{ error }}</span>
        {% endfor %}
    </div>
    <div class="form-group">
        {{ form.body.label }}<br>
        {{ form.body(rows=15, cols=80) }}<br>
        {% for error in form.body.errors %}
            <span class="error">{{ error }}</span>
        {% endfor %}
    </div>
    <div class="form-group">
        {{ form.submit() }}
    </div>
</form>
{% endblock %}
templates/main/edit_post.html
html 复制代码
{% extends "base.html" %}

{% block title %}编辑文章{% endblock %}

{% block content %}
<h2>编辑文章</h2>
<form method="POST">
    {{ form.hidden_tag() }}
    <div class="form-group">
        {{ form.title.label }}<br>
        {{ form.title(size=64) }}<br>
        {% for error in form.title.errors %}
            <span class="error">{{ error }}</span>
        {% endfor %}
    </div>
    <div class="form-group">
        {{ form.body.label }}<br>
        {{ form.body(rows=15, cols=80) }}<br>
        {% for error in form.body.errors %}
            <span class="error">{{ error }}</span>
        {% endfor %}
    </div>
    <div class="form-group">
        {{ form.submit() }}
    </div>
</form>
{% endblock %}

13. 静态文件 static/style.css(简单示例)

css 复制代码
body {
    font-family: Arial, sans-serif;
    margin: 0;
    padding: 0;
    background: #f4f4f4;
}

.container {
    width: 80%;
    margin: auto;
    overflow: hidden;
}

nav {
    background: #333;
    color: #fff;
    padding: 10px 0;
}

nav .container {
    display: flex;
    justify-content: space-between;
    align-items: center;
}

nav a {
    color: #fff;
    text-decoration: none;
}

nav .brand {
    font-size: 1.5em;
    font-weight: bold;
}

nav ul {
    list-style: none;
    margin: 0;
    padding: 0;
    display: flex;
}

nav ul li {
    margin-left: 20px;
}

main {
    background: #fff;
    padding: 20px;
    margin-top: 20px;
    margin-bottom: 20px;
    border-radius: 5px;
}

footer {
    background: #333;
    color: #fff;
    text-align: center;
    padding: 10px 0;
    margin-top: 20px;
}

.post {
    border-bottom: 1px solid #ccc;
    margin-bottom: 20px;
    padding-bottom: 10px;
}

.post h2 {
    margin-bottom: 5px;
}

.post-meta {
    color: #777;
    font-size: 0.9em;
    margin-bottom: 10px;
}

.post-summary {
    margin-bottom: 10px;
}

.post-actions {
    margin-top: 10px;
}

.post-actions a, .post-actions button {
    margin-right: 10px;
    background: #333;
    color: #fff;
    border: none;
    padding: 5px 10px;
    cursor: pointer;
    text-decoration: none;
    display: inline-block;
}

.post-actions button:hover, .post-actions a:hover {
    background: #555;
}

.pagination {
    text-align: center;
    margin-top: 20px;
}

.pagination a, .pagination span {
    margin: 0 5px;
}

.form-group {
    margin-bottom: 15px;
}

.form-group label {
    display: block;
    margin-bottom: 5px;
}

.form-group input, .form-group textarea {
    width: 100%;
    padding: 8px;
    border: 1px solid #ccc;
    border-radius: 4px;
}

.form-group input[type="submit"] {
    width: auto;
    background: #333;
    color: #fff;
    border: none;
    cursor: pointer;
}

.form-group input[type="submit"]:hover {
    background: #555;
}

.error {
    color: red;
    font-size: 0.9em;
}

.alert {
    padding: 10px;
    margin-bottom: 15px;
    border-radius: 4px;
}

.alert-success {
    background: #d4edda;
    color: #155724;
    border: 1px solid #c3e6cb;
}

.alert-danger {
    background: #f8d7da;
    color: #721c24;
    border: 1px solid #f5c6cb;
}

14. 启动入口 run.py

python 复制代码
from app import create_app

app = create_app()

if __name__ == '__main__':
    app.run(debug=True)

15. 环境变量示例 .env

复制代码
SECRET_KEY=your-secret-key-change-in-production
DATABASE_URL=sqlite:///blog.db
MAIL_SERVER=smtp.qq.com
MAIL_PORT=587
MAIL_USE_TLS=true
MAIL_USERNAME=your-email@qq.com
MAIL_PASSWORD=your-authorization-code
CACHE_TYPE=SimpleCache

16. 运行步骤

  1. 将以上所有文件按目录结构创建好。

  2. 安装依赖:pip install -r requirements.txt

  3. 初始化数据库:

    bash 复制代码
    flask db init
    flask db migrate -m "initial migration"
    flask db upgrade
  4. 设置环境变量(或使用 .env 文件):

    bash 复制代码
    export FLASK_APP=run.py
    export FLASK_ENV=development
  5. 启动应用:

    bash 复制代码
    flask run
  6. 访问 http://127.0.0.1:5000 使用。


三、总结与扩展建议

通过本系列教程,我们从零开始构建了一个功能完整的 Flask 博客系统,涵盖了 Flask 开发的主要知识点。这个系统可以作为你进一步学习的起点:

  • 添加评论功能:增加 Comment 模型,与 Post 关联。
  • 实现文章分类与标签:使用多对多关系。
  • 用户个人主页:展示用户发表的所有文章。
  • 富文本编辑器:集成 CKEditor 或 Markdown 编辑器。
  • RESTful API:使用 Flask-RESTx 提供 API 接口。
  • 单元测试:使用 pytest 编写测试用例。
  • 性能优化:使用 Redis 缓存、数据库索引等。

Flask 的生态非常丰富,你可以根据实际需求选择适合的扩展。希望本系列教程能帮助你打下坚实的基础,并在实际项目中灵活运用 Flask。

感谢阅读,祝你编码愉快!

相关推荐
观测云2 小时前
Python 应用实现 APM 自动注入(Kubernetes 篇)
开发语言·python·kubernetes
极光代码工作室2 小时前
基于Spark的用户行为分析系统设计
大数据·hadoop·python·数据分析·数据可视化
智算菩萨2 小时前
【Pygame】第9章 动画系统与帧动画
python·pygame
jinanwuhuaguo2 小时前
最新更新版本,OpenClaw v2026.4.2 深度解读剖析:Task Flow 重磅回归与安全架构的全面硬化
android·开发语言·人工智能·回归·kotlin·安全架构·openclaw
放飞自我的Coder2 小时前
【基于xGBoost的钓鱼邮件智能识别与拦截系统】
python
DFT计算杂谈2 小时前
eDMFT安装教程
java·服务器·前端·python·算法
小陈工2 小时前
2026年4月3日技术资讯洞察:微服务理性回归、AI代码生成争议与开源安全新挑战
开发语言·数据库·人工智能·python·安全·微服务·回归
CesareCheung2 小时前
Python+Vue +K6接口性能压测平台搭建
开发语言·vue.js·python
云烟成雨TD2 小时前
Spring AI 1.x 系列【23】:工具配置详解(全局默认+运行时动态)
人工智能·python·spring