经过前三篇的系统学习,我们已经掌握了 Flask 从入门到进阶的完整知识体系。本篇将对 Flask 的核心知识进行总结,并在此基础上提供一个完整的、可直接运行的简易博客系统,将所学内容融会贯通,帮助你巩固知识并快速应用于实际项目。
Flask 类的实例,是整个应用的入口。@app.route 装饰器将 URL 映射到视图函数,支持动态 URL 和多种 HTTP 方法。Response 对象等。request 对象封装了客户端发送的请求信息,如表单数据、查询参数、JSON 数据、请求头等。Response 对象,可以手动构建更复杂的响应。static 目录,通过 url_for('static', filename='...') 生成 URL。db.Model,使用 db.Column 定义字段,支持关系映射(一对多、多对多)。app.register_blueprint 注册到应用。url_prefix,实现路由分组。before_request、after_request、teardown_request下面我们将前面的知识点整合成一个功能完备的博客系统。该系统具有以下特性:
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
创建虚拟环境并安装依赖:
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
config.pyimport os
from dotenv import load_dotenv
load_dotenv()
classConfig:
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')
app/__init__.pyfrom 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()
defcreate_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
app/models.pyfrom app import db
from flask_login import UserMixin
from werkzeug.security import generate_password_hash, check_password_hash
from datetime import datetime
classUser(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')
defset_password(self, password):
self.password_hash = generate_password_hash(password)
defcheck_password(self, password):
return check_password_hash(self.password_hash, password)
def__repr__(self):
returnf'<User {self.username}>'
classPost(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):
returnf'<Post {self.title}>'
app/forms.pyfrom flask_wtf import FlaskForm
from wtforms import StringField, PasswordField, SubmitField, TextAreaField
from wtforms.validators import DataRequired, Email, Length, EqualTo
classRegistrationForm(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('注册')
classLoginForm(FlaskForm):
email = StringField('邮箱', validators=[DataRequired(), Email()])
password = PasswordField('密码', validators=[DataRequired()])
submit = SubmitField('登录')
classPostForm(FlaskForm):
title = StringField('标题', validators=[DataRequired()])
body = TextAreaField('内容', validators=[DataRequired()])
submit = SubmitField('发布')
app/auth/__init__.pyfrom flask import Blueprint
auth_bp = Blueprint('auth', __name__, url_prefix='/auth')
from app.auth import routes
app/auth/routes.pyfrom 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'])
defregister():
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'])
deflogin():
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')
deflogout():
logout_user()
return redirect(url_for('main.index'))
app/main/__init__.pyfrom flask import Blueprint
main_bp = Blueprint('main', __name__)
from app.main import routes
app/main/routes.pyfrom 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)
defindex():
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>')
defpost_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
defcreate_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
defedit_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
defdelete_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'))
app/__init__.py 中补充)from app.models import User
@login_manager.user_loader
defload_user(user_id):
return User.query.get(int(user_id))
将这段代码放在 create_app 函数中,login_manager.init_app(app) 之后。
templates/base.html<!DOCTYPEhtml>
<htmllang="zh">
<head>
<metacharset="UTF-8">
<metaname="viewport"content="width=device-width, initial-scale=1.0">
<title>{% block title %}博客系统{% endblock %}</title>
<linkrel="stylesheet"href="{{ url_for('static', filename='style.css') }}">
</head>
<body>
<nav>
<divclass="container">
<ahref="{{ url_for('main.index') }}"class="brand">我的博客</a>
<ulclass="nav-links">
{% if current_user.is_authenticated %}
<li>你好,{{ current_user.username }}</li>
<li><ahref="{{ url_for('main.create_post') }}">写文章</a></li>
<li><ahref="{{ url_for('auth.logout') }}">登出</a></li>
{% else %}
<li><ahref="{{ url_for('auth.login') }}">登录</a></li>
<li><ahref="{{ url_for('auth.register') }}">注册</a></li>
{% endif %}
</ul>
</div>
</nav>
<mainclass="container">
{% with messages = get_flashed_messages(with_categories=true) %}
{% if messages %}
{% for category, message in messages %}
<divclass="alert alert-{{ category }}">{{ message }}</div>
{% endfor %}
{% endif %}
{% endwith %}
{% block content %}{% endblock %}
</main>
<footer>
<divclass="container">
<p>© 2023 我的博客</p>
</div>
</footer>
</body>
</html>
templates/auth/register.html{% extends "base.html" %}
{% block title %}注册{% endblock %}
{% block content %}
<h2>注册</h2>
<formmethod="POST">
{{ form.hidden_tag() }}
<divclass="form-group">
{{ form.username.label }}<br>
{{ form.username(size=32) }}<br>
{% for error in form.username.errors %}
<spanclass="error">{{ error }}</span>
{% endfor %}
</div>
<divclass="form-group">
{{ form.email.label }}<br>
{{ form.email(size=32) }}<br>
{% for error in form.email.errors %}
<spanclass="error">{{ error }}</span>
{% endfor %}
</div>
<divclass="form-group">
{{ form.password.label }}<br>
{{ form.password(size=32) }}<br>
{% for error in form.password.errors %}
<spanclass="error">{{ error }}</span>
{% endfor %}
</div>
<divclass="form-group">
{{ form.confirm.label }}<br>
{{ form.confirm(size=32) }}<br>
{% for error in form.confirm.errors %}
<spanclass="error">{{ error }}</span>
{% endfor %}
</div>
<divclass="form-group">
{{ form.submit() }}
</div>
</form>
{% endblock %}
templates/auth/login.html{% extends "base.html" %}
{% block title %}登录{% endblock %}
{% block content %}
<h2>登录</h2>
<formmethod="POST">
{{ form.hidden_tag() }}
<divclass="form-group">
{{ form.email.label }}<br>
{{ form.email(size=32) }}<br>
{% for error in form.email.errors %}
<spanclass="error">{{ error }}</span>
{% endfor %}
</div>
<divclass="form-group">
{{ form.password.label }}<br>
{{ form.password(size=32) }}<br>
{% for error in form.password.errors %}
<spanclass="error">{{ error }}</span>
{% endfor %}
</div>
<divclass="form-group">
{{ form.submit() }}
</div>
</form>
{% endblock %}
templates/main/index.html{% extends "base.html" %}
{% block title %}首页{% endblock %}
{% block content %}
<h1>最新文章</h1>
{% for post in posts.items %}
<articleclass="post">
<h2><ahref="{{ url_for('main.post_detail', id=post.id) }}">{{ post.title }}</a></h2>
<divclass="post-meta">
作者:{{ post.author.username }} | 发布于:{{ post.created_at.strftime('%Y-%m-%d') }}
</div>
<divclass="post-summary">
{{ post.body[:200] }}{% if post.body|length > 200 %}...{% endif %}
</div>
{% if current_user == post.author %}
<divclass="post-actions">
<ahref="{{ url_for('main.edit_post', id=post.id) }}">编辑</a>
<formaction="{{ url_for('main.delete_post', id=post.id) }}"method="POST"style="display:inline;">
<buttontype="submit"onclick="returnconfirm('确定删除吗?')">删除</button>
</form>
</div>
{% endif %}
</article>
{% else %}
<p>暂无文章</p>
{% endfor %}
<divclass="pagination">
{% if posts.has_prev %}
<ahref="{{ url_for('main.index', page=posts.prev_num) }}">上一页</a>
{% endif %}
<span>第 {{ posts.page }} 页 / 共 {{ posts.pages }} 页</span>
{% if posts.has_next %}
<ahref="{{ url_for('main.index', page=posts.next_num) }}">下一页</a>
{% endif %}
</div>
{% endblock %}
templates/main/post_detail.html{% extends "base.html" %}
{% block title %}{{ post.title }}{% endblock %}
{% block content %}
<articleclass="post-detail">
<h1>{{ post.title }}</h1>
<divclass="post-meta">
作者:{{ post.author.username }} | 发布于:{{ post.created_at.strftime('%Y-%m-%d %H:%M') }}
</div>
<divclass="post-body">
{{ post.body|safe }}
</div>
{% if current_user == post.author %}
<divclass="post-actions">
<ahref="{{ url_for('main.edit_post', id=post.id) }}">编辑</a>
<formaction="{{ url_for('main.delete_post', id=post.id) }}"method="POST"style="display:inline;">
<buttontype="submit"onclick="returnconfirm('确定删除吗?')">删除</button>
</form>
</div>
{% endif %}
</article>
{% endblock %}
templates/main/create_post.html{% extends "base.html" %}
{% block title %}写文章{% endblock %}
{% block content %}
<h2>发布新文章</h2>
<formmethod="POST">
{{ form.hidden_tag() }}
<divclass="form-group">
{{ form.title.label }}<br>
{{ form.title(size=64) }}<br>
{% for error in form.title.errors %}
<spanclass="error">{{ error }}</span>
{% endfor %}
</div>
<divclass="form-group">
{{ form.body.label }}<br>
{{ form.body(rows=15, cols=80) }}<br>
{% for error in form.body.errors %}
<spanclass="error">{{ error }}</span>
{% endfor %}
</div>
<divclass="form-group">
{{ form.submit() }}
</div>
</form>
{% endblock %}
templates/main/edit_post.html{% extends "base.html" %}
{% block title %}编辑文章{% endblock %}
{% block content %}
<h2>编辑文章</h2>
<formmethod="POST">
{{ form.hidden_tag() }}
<divclass="form-group">
{{ form.title.label }}<br>
{{ form.title(size=64) }}<br>
{% for error in form.title.errors %}
<spanclass="error">{{ error }}</span>
{% endfor %}
</div>
<divclass="form-group">
{{ form.body.label }}<br>
{{ form.body(rows=15, cols=80) }}<br>
{% for error in form.body.errors %}
<spanclass="error">{{ error }}</span>
{% endfor %}
</div>
<divclass="form-group">
{{ form.submit() }}
</div>
</form>
{% endblock %}
static/style.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;
}
run.pyfrom app import create_app
app = create_app()
if __name__ =='__main__':
app.run(debug=True)
.envSECRET_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
pip install -r requirements.txtflask db init
flask db migrate -m"initial migration"
flask db upgrade
.env 文件):exportFLASK_APP=run.py
exportFLASK_ENV=development
flask run
http://127.0.0.1:5000 使用。通过本系列教程,我们从零开始构建了一个功能完整的 Flask 博客系统,涵盖了 Flask 开发的主要知识点。这个系统可以作为你进一步学习的起点:
Flask 的生态非常丰富,你可以根据实际需求选择适合的扩展。希望本系列教程能帮助你打下坚实的基础,并在实际项目中灵活运用 Flask。
欢迎关注公众号,感谢对文章的点赞分享喜欢,冉成未来会持续更新前后端开发技术、人工智能技术、IT相关的文章及学习经验、知识分享,未来虽然充满着不确定性,但我们可以不断提升自己,不断为未来做准备,让未来更好的自己成就更美好的未来。
Python基础系列 | Python之PyQt5基础知识(五)
Python基础系列 | Python之PyQt5基础知识(四)
Python基础系列 | Python之PyQt5基础知识(三)
Python基础系列 | Python之PyQt5基础知识(二)
Python基础系列 | Python之PyQt5基础知识(一)