{{ post.title }}
作者: {{ post.author.username }} | 发布于: {{ post.timestamp.strftime('%Y-%m-%d %H:%M') }} {% if current_user == post.author %} 编辑 {% endif %}
在Python Web开发领域,Flask以其轻量级、灵活性和易用性赢得了众多开发者的青睐。本文将带您从零开始构建一个功能完善的博客系统,包含文章发布、评论互动和标签分类等核心功能。通过这个实战项目,您不仅能掌握Flask的核心技术,还能学习到现代Web开发的最佳实践。
我们的博客系统将包含以下核心功能模块:
用户认证(注册/登录/登出)
博客文章管理(创建/编辑/删除)
评论系统(文章评论/回复)
标签分类(多标签关联)
文章分页与搜索
首先创建项目环境:
# 创建项目目录
mkdir flask-blog && cd flask-blog
# 创建虚拟环境
python -m venv venv
# 激活虚拟环境
# Windows: venv\Scripts\activate
# Mac/Linux: source venv/bin/activate
# 安装Flask及其他依赖
pip install flask flask-sqlalchemy flask-login flask-wtf flask-migrate
合理的项目结构是良好开端:
flask-blog/
│
├── app/
│ ├── __init__.py # 应用工厂函数
│ ├── models.py # 数据模型
│ ├── routes.py # 路由定义
│ ├── forms.py # 表单类
│ ├── templates/ # 模板文件
│ │ ├── base.html # 基础模板
│ │ ├── auth/ # 认证相关模板
│ │ └── blog/ # 博客相关模板
│ └── static/ # 静态文件
│
├── migrations/ # 数据库迁移脚本
├── config.py # 配置文件
└── run.py # 启动脚本
在app/models.py
中定义我们的数据模型:
from datetime import datetime
from app import db, login_manager
from flask_login import UserMixin
from werkzeug.security import generate_password_hash, check_password_hash
# 用户模型
class User(UserMixin, db.Model):
id = db.Column(db.Integer, primary_key=True)
username = db.Column(db.String(64), index=True, unique=True)
email = db.Column(db.String(120), index=True, unique=True)
password_hash = db.Column(db.String(128))
posts = db.relationship('Post', backref='author', lazy='dynamic')
comments = db.relationship('Comment', 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)
# 文章模型
class Post(db.Model):
id = db.Column(db.Integer, primary_key=True)
title = db.Column(db.String(140))
content = db.Column(db.Text)
timestamp = db.Column(db.DateTime, index=True, default=datetime.utcnow)
user_id = db.Column(db.Integer, db.ForeignKey('user.id'))
comments = db.relationship('Comment', backref='post', lazy='dynamic')
tags = db.relationship('Tag', secondary='post_tag', backref=db.backref('posts', lazy='dynamic'))
# 评论模型
class Comment(db.Model):
id = db.Column(db.Integer, primary_key=True)
content = db.Column(db.Text)
timestamp = db.Column(db.DateTime, index=True, default=datetime.utcnow)
user_id = db.Column(db.Integer, db.ForeignKey('user.id'))
post_id = db.Column(db.Integer, db.ForeignKey('post.id'))
parent_id = db.Column(db.Integer, db.ForeignKey('comment.id'))
replies = db.relationship('Comment', backref=db.backref('parent', remote_side=[id]), lazy='dynamic')
# 标签模型
class Tag(db.Model):
id = db.Column(db.Integer, primary_key=True)
name = db.Column(db.String(50), unique=True)
# 文章-标签关联表
post_tag = db.Table('post_tag',
db.Column('post_id', db.Integer, db.ForeignKey('post.id')),
db.Column('tag_id', db.Integer, db.ForeignKey('tag.id'))
)
@login_manager.user_loader
def load_user(id):
return User.query.get(int(id))
实现用户注册、登录和登出功能:
# app/forms.py
from flask_wtf import FlaskForm
from wtforms import StringField, PasswordField, SubmitField, TextAreaField
from wtforms.validators import DataRequired, Email, EqualTo, Length
class LoginForm(FlaskForm):
username = StringField('用户名', validators=[DataRequired()])
password = PasswordField('密码', validators=[DataRequired()])
submit = SubmitField('登录')
class RegistrationForm(FlaskForm):
username = StringField('用户名', validators=[DataRequired(), Length(min=4, max=25)])
email = StringField('邮箱', validators=[DataRequired(), Email()])
password = PasswordField('密码', validators=[DataRequired()])
password2 = PasswordField('确认密码', validators=[DataRequired(), EqualTo('password')])
submit = SubmitField('注册')
# app/routes.py
from flask import render_template, flash, redirect, url_for, request
from app import app, db
from app.forms import LoginForm, RegistrationForm
from app.models import User
from flask_login import current_user, login_user, logout_user, login_required
@app.route('/login', methods=['GET', 'POST'])
def login():
if current_user.is_authenticated:
return redirect(url_for('index'))
form = LoginForm()
if form.validate_on_submit():
user = User.query.filter_by(username=form.username.data).first()
if user is None or not user.check_password(form.password.data):
flash('无效的用户名或密码')
return redirect(url_for('login'))
login_user(user, remember=form.remember_me.data)
next_page = request.args.get('next')
return redirect(next_page) if next_page else redirect(url_for('index'))
return render_template('auth/login.html', title='登录', form=form)
@app.route('/register', methods=['GET', 'POST'])
def register():
if current_user.is_authenticated:
return redirect(url_for('index'))
form = RegistrationForm()
if form.validate_on_submit():
user = User(username=form.username.data, email=form.email.data)
user.set_password(form.password.data)
db.session.add(user)
db.session.commit()
flash('恭喜,注册成功!')
return redirect(url_for('login'))
return render_template('auth/register.html', title='注册', form=form)
@app.route('/logout')
def logout():
logout_user()
return redirect(url_for('index'))
实现文章的增删改查功能:
# app/forms.py
class PostForm(FlaskForm):
title = StringField('标题', validators=[DataRequired(), Length(max=140)])
content = TextAreaField('内容', validators=[DataRequired()])
tags = StringField('标签(用逗号分隔)')
submit = SubmitField('发布')
# app/routes.py
from app.forms import PostForm
from app.models import Post, Tag
@app.route('/', methods=['GET', 'POST'])
@app.route('/index', methods=['GET', 'POST'])
@login_required
def index():
form = PostForm()
if form.validate_on_submit():
post = Post(title=form.title.data, content=form.content.data, author=current_user)
# 处理标签
if form.tags.data:
tag_names = [name.strip() for name in form.tags.data.split(',')]
for name in tag_names:
tag = Tag.query.filter_by(name=name).first()
if tag is None:
tag = Tag(name=name)
db.session.add(tag)
post.tags.append(tag)
db.session.add(post)
db.session.commit()
flash('您的文章已发布!')
return redirect(url_for('index'))
page = request.args.get('page', 1, type=int)
posts = Post.query.order_by(Post.timestamp.desc()).paginate(
page, app.config['POSTS_PER_PAGE'], False)
return render_template('blog/index.html', title='首页', form=form, posts=posts)
@app.route('/post/')
def post(post_id):
post = Post.query.get_or_404(post_id)
return render_template('blog/post.html', post=post)
@app.route('/edit/', methods=['GET', 'POST'])
@login_required
def edit_post(post_id):
post = Post.query.get_or_404(post_id)
if post.author != current_user:
abort(403)
form = PostForm()
if form.validate_on_submit():
post.title = form.title.data
post.content = form.content.data
# 更新标签
post.tags = []
if form.tags.data:
tag_names = [name.strip() for name in form.tags.data.split(',')]
for name in tag_names:
tag = Tag.query.filter_by(name=name).first()
if tag is None:
tag = Tag(name=name)
db.session.add(tag)
post.tags.append(tag)
db.session.commit()
flash('文章已更新')
return redirect(url_for('post', post_id=post.id))
elif request.method == 'GET':
form.title.data = post.title
form.content.data = post.content
form.tags.data = ', '.join([tag.name for tag in post.tags])
return render_template('blog/edit_post.html', title='编辑文章', form=form)
@app.route('/delete/', methods=['POST'])
@login_required
def delete_post(post_id):
post = Post.query.get_or_404(post_id)
if post.author != current_user:
abort(403)
db.session.delete(post)
db.session.commit()
flash('文章已删除')
return redirect(url_for('index'))
实现多级评论功能:
# app/forms.py
class CommentForm(FlaskForm):
content = TextAreaField('评论内容', validators=[DataRequired()])
submit = SubmitField('提交')
# app/routes.py
from app.forms import CommentForm
from app.models import Comment
@app.route('/post/', methods=['GET', 'POST'])
def post(post_id):
post = Post.query.get_or_404(post_id)
form = CommentForm()
if form.validate_on_submit():
if not current_user.is_authenticated:
flash('请先登录再评论')
return redirect(url_for('login'))
comment = Comment(content=form.content.data,
author=current_user,
post=post)
db.session.add(comment)
db.session.commit()
flash('您的评论已发布')
return redirect(url_for('post', post_id=post.id))
# 获取顶级评论(非回复的评论)
top_level_comments = Comment.query.filter_by(post_id=post.id, parent_id=None)\
.order_by(Comment.timestamp.desc()).all()
return render_template('blog/post.html', post=post, form=form,
comments=top_level_comments)
@app.route('/reply/', methods=['POST'])
@login_required
def reply(comment_id):
parent_comment = Comment.query.get_or_404(comment_id)
post = parent_comment.post
form = CommentForm()
if form.validate_on_submit():
comment = Comment(content=form.content.data,
author=current_user,
post=post,
parent=parent_comment)
db.session.add(comment)
db.session.commit()
flash('您的回复已发布')
return redirect(url_for('post', post_id=post.id))
添加标签相关的视图函数:
@app.route('/tag/')
def tag(tag_name):
tag = Tag.query.filter_by(name=tag_name).first_or_404()
page = request.args.get('page', 1, type=int)
posts = tag.posts.order_by(Post.timestamp.desc()).paginate(
page, app.config['POSTS_PER_PAGE'], False)
return render_template('blog/tag.html', tag=tag, posts=posts)
@app.route('/tags')
def tags():
all_tags = Tag.query.order_by(Tag.name).all()
return render_template('blog/tags.html', tags=all_tags)
{% block title %}{% endblock %} - Flask博客
{% with messages = get_flashed_messages(with_categories=true) %}
{% if messages %}
{% for category, message in messages %}
{{ message }}
{% endfor %}
{% endif %}
{% endwith %}
{% block content %}{% endblock %}
{% block scripts %}{% endblock %}
{% extends "base.html" %}
{% block content %}
最新文章
{% if current_user.is_authenticated %}
发表新文章
{% endif %}
{% for post in posts.items %}
{{ post.title }}
作者: {{ post.author.username }} |
发布于: {{ post.timestamp.strftime('%Y-%m-%d %H:%M') }}
{{ post.content[:200] }}...
{% for tag in post.tags %}
{{ tag.name }}
{% endfor %}
阅读全文 →
{% endfor %}
热门标签
{% for tag in Tag.query.order_by(Tag.name).limit(20).all() %}
{{ tag.name }}
{% endfor %}
{% endblock %}
{% extends "base.html" %}
{% block content %}
{{ post.title }}
作者: {{ post.author.username }} |
发布于: {{ post.timestamp.strftime('%Y-%m-%d %H:%M') }}
{% if current_user == post.author %}
编辑
{% endif %}
{% for tag in post.tags %}
{{ tag.name }}
{% endfor %}
{{ post.content | safe }}
评论 ({{ post.comments.count() }})
{% if current_user.is_authenticated %}
{% else %}
登录后发表评论
{% endif %}
{% for comment in comments %}
{% include '_comment.html' %}
{% endfor %}
{% endblock %}
{{ comment.author.username }}
{{ comment.timestamp.strftime('%Y-%m-%d %H:%M') }}
{% if current_user.is_authenticated %}
{% endif %}
{{ comment.content }}
{% for reply in comment.replies.order_by(Comment.timestamp.asc()) %}
{{ reply.author.username }}
{{ reply.timestamp.strftime('%Y-%m-%d %H:%M') }}
{{ reply.content }}
{% endfor %}
{% block scripts %}
{% endblock %}
创建config.py
配置文件:
import os
from dotenv import load_dotenv
basedir = os.path.abspath(os.path.dirname(__file__))
load_dotenv(os.path.join(basedir, '.env'))
class Config:
SECRET_KEY = os.environ.get('SECRET_KEY') or 'dev-secret-key'
SQLALCHEMY_DATABASE_URI = os.environ.get('DATABASE_URL') or \
'sqlite:///' + os.path.join(basedir, 'app.db')
SQLALCHEMY_TRACK_MODIFICATIONS = False
POSTS_PER_PAGE = 10
class ProductionConfig(Config):
pass
class DevelopmentConfig(Config):
DEBUG = True
config = {
'development': DevelopmentConfig,
'production': ProductionConfig,
'default': DevelopmentConfig
}
安装Gunicorn:
pip install gunicorn
创建启动脚本wsgi.py
:
from app import create_app
app = create_app()
if __name__ == '__main__':
app.run()
启动命令:
gunicorn -w 4 -b 0.0.0.0:8000 wsgi:app
数据库优化:
为常用查询字段添加索引
使用Flask-SQLAlchemy的get()
代替filter_by().first()
合理使用lazy
加载策略
缓存策略:
使用Flask-Caching缓存频繁访问的数据
实现文章浏览计数器的延迟更新
静态文件处理:
配置Nginx直接处理静态文件
使用CDN分发静态资源
启用Gzip压缩
用户个人中心:
头像上传功能
个人资料编辑
用户关注系统
增强搜索功能:
实现全文搜索(Elasticsearch或Whoosh)
添加搜索建议和自动完成
社交功能:
文章点赞/收藏
用户私信系统
通知系统
API开发:
使用Flask-RESTful开发RESTful API
实现前后端分离架构
通过本教程,我们完成了一个功能完善的Flask博客系统,包含了用户认证、文章管理、评论系统和标签分类等核心功能。这个项目不仅展示了Flask的核心技术,也体现了现代Web开发的最佳实践。
希望这个项目能作为您Flask学习之旅的良好起点。您可以根据自己的需求继续扩展功能,比如添加用户头像、实现文章搜索、开发RESTful API等。
如果您在实现过程中遇到任何问题,或者有改进建议,欢迎在评论区留言讨论!