当前位置: 首页 > news >正文

Flask数据库迁移实战指南

数据库迁移操作

1. 安装必要的依赖

pip install flask flask-sqlalchemy flask-migrate pymysql cryptography

2. 创建项目结构

# myapp/
# ├── app.py          # 主应用文件
# ├── models.py       # 数据模型定义
# └── config.py       # 配置文件

3. config.py (数据库配置)

class Config:# MySQL 数据库连接配置 - 请替换为你的实际数据库信息SQLALCHEMY_DATABASE_URI = 'mysql+pymysql://username:password@localhost:3306/flask_db'SQLALCHEMY_TRACK_MODIFICATIONS = False  # 避免警告信息MYSQL_DEFAULT_CHARSET = 'utf8mb4'      # 支持完整UnicodeMYSQL_DEFAULT_COLLATION = 'utf8mb4_unicode_ci'  # 排序规则

4. app.py (主应用文件)

from flask import Flask
from flask_sqlalchemy import SQLAlchemy
from flask_migrate import Migrate
​
class Config:# MySQL 数据库连接配置 - 请替换为你的实际数据库信息SQLALCHEMY_DATABASE_URI = 'mysql://root:123456@127.0.0.1:3306/youdb'SQLALCHEMY_TRACK_MODIFICATIONS = False  # 避免警告信息
​
app = Flask(__name__)
app.config.from_object(Config)  # 加载配置
​
# 初始化数据库扩展
db = SQLAlchemy(app)
migrate = Migrate(app, db)  # 初始化迁移扩展
​
# 注意:导入模型必须在 db 初始化后
from models import User, Post

5. models.py (数据模型定义)

from app import db
from datetime import datetime
​
class User(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), nullable=False)created_at = db.Column(db.DateTime, default=datetime.utcnow)# 一对多关系: 一个用户有多篇文章posts = db.relationship('Post', backref='author', lazy='dynamic')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(120), nullable=False)content = 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'), nullable=False)def __repr__(self):return f'<Post {self.title}>'

6. 初始化迁移环境

  • 在终端里面运行

set FLASK_APP=app.py
flask db init   #进行初始化
​
# 结果: 在当前工作目录中会创建 migrations 目录

7. 创建初始迁移脚本 (检测模型变化)

flask db migrate -m "第一次操作"
​
# 结果: 
# - 在 migrations/versions 生成迁移脚本文件 (如: 1a2b3c4d5e6f_initial_migration.py)
# - 检查生成的脚本文件是否准确反映了模型定义

8. 应用迁移到 MySQL 数据库

# 步骤 5: 应用迁移到 MySQL 数据库
flask db upgrade
​
# 结果:
# 1. 创建 youdb 数据库(若不存在)
# 2. 创建 User 和 Post 表
# 3. 在 alembic_version 表中记录迁移版本

9. 修改模型 (添加新字段)

# 文件: models.py (更新模型)
​
class User(db.Model):# ... 原有字段 ...# 新增字段is_admin = db.Column(db.Boolean, default=False)  # 管理标识last_login = db.Column(db.DateTime)              # 最后登录时间

10. 生成新的迁移脚本

flask db migrate -m "第二次操作,添加了新的user字段"
​
# 生成的迁移脚本示例 (migrations/versions/xxx_add_fields.py):
"""
def upgrade():# ### commands auto generated by Alembic - please adjust! ###op.add_column('user', sa.Column('is_admin', sa.Boolean(), nullable=True))op.add_column('user', sa.Column('last_login', sa.DateTime(), nullable=True))# ### end Alembic commands #### 自定义操作: 设置默认值op.execute("UPDATE user SET is_admin = 0")  # MySQL使用0表示False
"""
​
# 注意事项:
# 1. 检查生成的脚本是否准确
# 2. 布尔类型在MySQL中实际是TINYINT(1)
# 3. 可能需要手动添加默认值逻辑

11. 应用新的迁移

flask db upgrade
​
# 结果: 
# User 表新增 is_admin 和 last_login 列

12. 回滚迁移操作

# 步骤 9: 回滚迁移 (如果需要)
# 查看迁移历史
flask db history
​
# 示例输出:
# 1a2b3c4d5e6f -> 7g8h9i0j1k2l (head), Add is_admin and last_login
# <base> -> 1a2b3c4d5e6f, Initial migration
​
# 回滚到上一个版本
flask db downgrade 1a2b3c4d5e6f  # 使用版本ID
​
# 结果: 
# 1. 删除 is_admin 和 last_login 列
# 2. 数据库回到之前的版本

13. 生产环境部署迁移

# 最佳实践:
# 1. 保留所有迁移脚本文件
# 2. 部署新代码前执行:
flask db upgrade
​
# 3. 回滚操作:
flask db downgrade <target-revision>

14. 常见问题解决方案

# 问题 1: 迁移失败 (外键约束)
# 解决方案: 在迁移脚本中添加依赖关系
down_revision = 'previous_version_id'  # 必须正确设置前一个版本ID
​
# 问题 2: MySQL 字符集问题
# 解决方案: 在数据库URI中指定字符集
SQLALCHEMY_DATABASE_URI = 'mysql+pymysql://user:pass@localhost/db?charset=utf8mb4'
​
# 问题 3: 长字段索引问题
# MySQL 对于长字符串创建索引有限制,解决方案:
username = db.Column(db.String(191), unique=True)  # 最大191字符可被索引

http://www.dtcms.com/a/344390.html

相关文章:

  • Flask电影投票系统全解析
  • 近期https接口的联调小记
  • STM32——SPI通信+W25Q64
  • 一体化伺服电机在特种机器人(炉管爬行器)中的应用案例
  • ShimetaPi M4-R1 :OpenHarmony 开发板解析
  • Mysql事务特性及原理
  • 网络安全基础知识
  • 异步开发的三种实现方式
  • 香港券商櫃台系統跨境金融研究
  • CTFshow系列——命令执行web45-48
  • 优选算法1:双指针
  • 如何在Vscode中配置MCP服务?(包含实例:使用Github Copilot + 高德MCP查询旅游攻略)
  • 聚焦AI与绿色双碳 金士顿亮相2025深圳国际电子展
  • 【链表 - LeetCode】2. 两数相加
  • 深度学习——神经网络
  • 深度学习赋能光纤非线性光学:Nature Comms揭示噪声驱动系统的智能预测框架
  • 【openGLES】着色器语言(GLSL)
  • CAM可视化卷积神经网络
  • 开源 python 应用 开发(十一)短语音转文本
  • 安卓手机格式转换,支持PDF转Word、PDF转Excel、PDF转PPT、PDT转图片
  • 基于ERNIE 4.5的多智能体协作的自动化视频舆情分析报告生成器
  • UE5基本打光(新手向)
  • vue2整合uniapp、uviewUi小程序开发
  • Yapi中通过MongoDB修改管理员密码与新增管理员
  • PDF 转 TIFF 性能测评:IronPDF具有更快的处理速度、更少的内存
  • 携程旅游的 AI 网关落地实践
  • 爬虫基础学习-链接协议分析,熟悉相关函数
  • C++中的右值引用与通用引用:std::move与std::forward的正确使用 (Effective Modern C++ 条款25)
  • 中项-基础知识分享12-软件工程
  • 保护 PDF 格式:禁止转换为其他格式文件