最近我在为 openEuler 社区添加一个 FAQ 模块, 这一系列正是我在这一过程中的总结

全部内容: 如何编写一个 Python Web 应用(零)


简单增删改查

参考:

复杂关系: 一对一, 一对多

在 Mybatis 中, 实现一对多和一堆一, 需要在 .xml 文件或 @Select 注解中生命映射, 但咱 SQLAlchemy 中, 只需要一个参数.

这是一个例子. 这里 EQuestionEQuestionDescription 是一对多的关系.

/models.py:

from faq import db


class EQuestion(db.Model):
    __tablename__ = 'e_question'

    id = db.Column(db.String(20), primary_key=True)
    std_description = db.Column(db.String(250), unique=True)


class EQuestionDescription(db.Model):
    __tablename__ = 'e_question_description'

    id = db.Column(db.String(20), primary_key=True)
    description = db.Column(db.String(200), nullable=False)
    question_id = db.Column(db.ForeignKey('e_question.id'), index=True)

    question = db.relationship('EQuestion', backref='e_question_descriptions', primaryjoin='EQuestionDescription.question_id == EQuestion.id')

只需要在 "一对多" 中 "多" 的一方加入:

question = db.relationship('EQuestion', backref='e_question_descriptions', primaryjoin='EQuestionDescription.question_id == EQuestion.id')

接下来, 编写一个 CLI COmmand 来测试一下:

@app.cli.command('hello')
def hello():
    question: EQuestion = EQuestion.query.get('2')
    print(question)

    descriptions = question.e_question_descriptions
    print(descriptions)

    description: EQuestionDescription = descriptions[0]
    print(description)

    question: EQuestion = description.question
    print(question)

结果:

PS D:\my\flask\app> flask hello
<EQuestion 2>
[<EQuestionDescription 3>, <EQuestionDescription 9>]
<EQuestionDescription 3>
<EQuestion 2>

使用 sqlacodegen 实现针对 SQLAlchemy 的逆向工程

这里只是介绍这个有力的工具:

参考: flask 插件 flask-sqlacodegen

需要注意的是, 由 flask-sqlacodegen 直接生成的代码并不能直接适应我们原有的代码

下面是在 PowerShell 中完整的生成命令. 他将在当前目录中生成一个 models.py 文件

PS D:\my\flask\app> (.\venv\Scripts\Activate.ps1) ;((flask-sqlacodegen mysql://root:123456@localhost:3306/openeuler_faq --flask) -replace 'db = SQLAlchemy\(\)','from faq import db' -replace 'from flask_sqlalchemy import SQLAlchemy','' | out-file faq/models.py -encoding utf8)

关于这一命令的含义, 我将在最后一节详谈

03-05 23:38