Flask sqlalchemy一对多与多对一与一对一及多对多关系介绍

 更新时间:2022年09月06日 16:11:01   作者:weixin_42576837  
这篇文章主要介绍了Flask sqlalchemy一对多与多对一与一对一及多对多关系介绍,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

这里以作者和文章来演示一对多的关系:一个作者可以有多篇文章,但是一篇文章只能有一个作者。

配置项

首先,配置下数据库config.py

username = 'xxxx'
password = 'xxxx'
database = 'school'
hostname = 'localhost'
port = '3306'
uri = f'mysql+pymysql://{username}:{password}@{hostname}:{port}/{database}'
SQLALCHEMY_DATABASE_URI  = uri
SQLALCHEMY_TRACK_MODIFICATIONS = False

在app.py文件中导入配置

from flask import Flask
from flask_sqlalchemy import SQLAlchemy
import config
app = Flask(__name__)
app.config.from_object(config)
db = SQLAlchemy(app)

接着创建模型类,建立python类到数据表的映射:

定义外键

class User(db.Model):
    __tablename__ = 'user'
    id = db.Column(db.Integer,primary_key=True,autoincrement=True)
    name = db.Column(db.String(100),nullable=False)
class Article(db.Model):
    __tablename__ = 'article'
    id = db.Column(db.Integer,primary_key=True,autoincrement=True)
    title = db.Column(db.String(200),nullable=False)
    content = db.Column(db.Text,nullable=False)
    user_id = db.Column(db.Integer,db.ForeignKey('user.id')) 

由于一个作者可以有多篇文章,所以外键应该设置在Article类中,这样每一篇文章的user_id字段都只会有一个值,因为只对应一个作者。

假设,在User类中存在外键字段article_id,那么一个作者的所有文章都需要存放在这一个字段中,但是外键只能存放单一数据(表量),所以外键的设置总是在“多”的这一侧定义。

定义关系属性

为什么需要关系属性,具体的原因我也不清楚,我想可能是从查询的角度来说,会更方便。

定义关系属性需要使用关系函数。关系属性在关系的出发侧定义,即一对多关系的“一”这一侧。一个作者拥有多篇文章,我们在User模型类中,定义一个叫articles的关系属性,用它可以表示每一个作者所对应的多篇文章。

class User(db.Model):
    __tablename__ = 'user'
    id = db.Column(db.Integer,primary_key=True,autoincrement=True)
    name = db.Column(db.String(100),nullable=False)
    articles = db.relationship('Article',backref=db.backref('user'))

articles 字段使用db.relationship()关系函数定义为关系属性,这个关系属性将返回多个记录。

  • relationship()函数的第一个参数为关系另一侧的模型名称,是python类的名称,不是数据表名称。
  • 第二个参数表示添加反向引用,会自动在另一侧,也就是Article模型中,建立一个关系属性,这个字段叫user,使用这个字段可以查找到文章所对应的用户。

现在user表中多了一个字段articles,但是它并不是数据库层面的,实际的表中并没有这个字段,可以认为只是一个查询接口。

接着创建数据表并插入数据:

 u1 = User(name='zs')
 a1 = Article(title='西游记',content='西游记是四大名著')
 a1.author = u1
 db.session.add(a1)
 db.session.commit()

创建了一个user表的记录,一个article表的记录,如何让他们建立联系呢?使用关系属性字段:

文章唯一指向一个作者:

a1.author = u1

直接将作者的实例对象u1赋值给文章实例对象a1的author字段,他们就会建立关系,文章表中的user_id字段就会指向那个作者。

article表

user表

现在用户zs拥有两篇文章,尝试使用关系属性查询。

u1 = User.query.filter_by(name='zs').first()
print(u1.articles)

输出user表的articles字段:

[<Article 1>, <Article 2>]

可以发现这个字段里面是两条记录,是article表中的两条记录,因为这两篇文章都是zs的文章,所以通过这个关系属性字段,可以获取到一个作者对应的所有文章。

反过来,关于反向引用,backref=db.backref('user'),它会在Article也建立一个关系属性,这个字段叫做user,可以通过这个字段获取到文章对应的作者。

比如:

#先找到文章
article1 = Article.query.filter_by(title='西游记').first()
#使用反向引用的字段,user,获取到这个文章对应的作者
print(article1.user)

输出:

<User 1>

双向的关系属性

上面在建立关系属性是,只是在User类中使用了关系属性:

articles = db.relationship('Article',backref=db.backref('user'))

这种方式,会隐式的在Article类中也建立一个关系属性user。我们可以使用back_populates参数显式的建立双向的关系属性。

这里仍然以作者author和文章article为例,一个作者可以有多篇文章,一篇文章只能有一个作者,建立显示的双向关系属性。

class User(db.Model):
    __tablename__ = 'author'
    id = db.Column(db.Integer,primary_key=True,autoincrement=True)
    name = db.Column(db.String(100),nullable=False)
    #关系属性
    articles = db.relationship('Article',back_populates='authors')
class Article(db.Model):
    __tablename__ = 'article'
    id = db.Column(db.Integer,primary_key=True,autoincrement=True)
    title = db.Column(db.String(200),nullable=False)
    content = db.Column(db.Text,nullable=False)
    auth_id = db.Column(db.Integer, db.ForeignKey('author.id'))
    #关系属性
    authors = db.relationship('User',back_populates='articles')

在一的一侧,User类中,建立了关系属性,articles,获取一个作者对应的多个文章记录。

在多的一侧,Article类中,建立了关系属性,authors,获取每一篇文章对应的一个作者记录。

使用back_populates参数连接对方,参数值需要设置为关系另一侧的关系属性名。

使用关系属性添加数据

u1 = User(name='zs')
a1 = Article(title='西游记', content='西游记是四大名著')
a2 = Article(title='红楼梦', content='红楼梦是四大名著')
a1.authors = u1
a2.authors = u1
db.session.add_all([a1,a2])
db.session.commit()

实例化User类对象u1,Article类对象a1,a2。

然后使用Article类的关系属性字段,authors

将User类对象u1赋值给Article类对象的关系属性authors

或者反过来,使用用户的关系属性字段articles添加数据:

def insert():
    u1 = User(name='zs')
    a1 = Article(title='西游记',content='西游记是四大名著')
    a2 = Article(title='水浒传',content='水浒传是四大名著')
    u1.articles = [a1,a2]
    db.session.add(u1)
    db.session.commit()

这里的添加方式是:u1.articles = [a1,a2]

接着查询,使用关系属性字段就可以查询到了,这里只是演示了使用back_populates参数显示的建立双向的关系属性,之前使用的backref可以简化关系的定义,是一种隐式的双向关系的建立。

一对一

这里使用国家和首都演示一对一关系:每一个国家只有一个首都;反过来说,一个城市也只能作为一个国家的首都。

一对一关系实际上是通过建立双向关系的一对多关系的基础上转化而来的。

class Country(db.Model):
    __tablename__ = 'country'
    id = db.Column(db.Integer,primary_key=True,autoincrement=True)
    name = db.Column(db.String(30),unique=True)
	capital = db.relationship('Capital',back_populates='country')
class Capital(db.Model):
    __tablename__ = 'capital'
    id = db.Column(db.Integer, primary_key=True, autoincrement=True)
    name = db.Column(db.String(30),unique=True)
    #外键在哪一个表中设置应该都可以
    country_id = db.Column(db.Integer,db.ForeignKey('country.id'))
    #设置关系属性
    country = db.relationship('Country',back_populates='capital')

首先上面是一个双向的关系属性,这时候表的关系和之前的一对多没有区别,现在我们需要给关系属性中假如一个参数uselist=False

”一“的一侧加入,就是在Country类模型中加入

capital = db.relationship('Capital',back_populates='country',uselist=False)

加入这个参数以后,在使用Country.capital获取记录时,将限制只返回一条记录。

由于Capital中设置了外键country_id ,存储单一数据,一个记录的country_id 只会对应Country中的一个记录。

这是一对多关系就变成了一对一关系。

测试一对一:

首先插入数据,插入一个国家记录,一个城市记录,然后建立关系。

def insert():
    china = Country(name='中国')
    Beijing = Capital(name='北京')
    Beijing.country = china
    db.session.add(Beijing)
    db.session.commit()

这样,Country表中就有了

capital表中:

他们目前时一对一的关系,那么,假如现在新增一个城市,它的country_id指向中国,

这时候就变成了一个国家,对应两个城市,变成了一对多,但是我们定义的是一对一,这样可行吗?

def f():
    #首先拿到中国的这条记录
    china = Country.query.filter_by(name='中国')
    china = china.first()
    print(china)
    #新增城市,
    Guangzhou = Capital(name='广州')
    #建立广州与中国之间的关系
    Guangzhou.country = china
    db.session.add(Guangzhou)
    db.session.commit()
    print('success')

执行这个函数之后,数据表会有以下变化:

可以看到capital表中的北京这条记录的country_id值变成了NULL,这正是因为我们建立的关系时一对一的,不允许变成一对多,所以,会把之前的对应关系取消掉。假如删除了参数uselist=False,就可以建立一对多的关系了。

多对多

这里使用学生和老师来演示多对多关系:每个学生有多个老师,每个老师也可以有多个学生。

在一对多关系中,我们可以在”多“这一侧添加外键指向”一“这一侧,外键只能存储一个记录,但是在多对多关系中,每一个记录都可以与关系另一侧的多个记录建立关系,关系两侧的模型都需要存储一组外键。

在SQLAlchemy中,要想表示多对多关系,除了关系两侧的模型外,我们还需要创建一个关联表(association table)。关联表不存储数据,只用来存储关系两侧模型的外键对应关系,

from flask import Flask
from flask_sqlalchemy import SQLAlchemy
import config
app = Flask(__name__)
app.config.from_object(config)
db = SQLAlchemy(app)
#关联表
association_table = db.Table('stu_tea_associ',#关联表的名称
    db.Column('student_id',db.Integer,db.ForeignKey('student.id')),#字段student_id,类型,关联的外键
    db.Column('teacher_id',db.Integer,db.ForeignKey('teacher.id')))#字段teacher_id,类型,关联的外键
#学生表
class Student(db.Model):
    __tablename__ = 'student'
    id = db.Column(db.Integer,primary_key=True,autoincrement=True)
    name = db.Column(db.String(20))
    #建立关系属性
    teachers = db.relationship('Teacher',secondary=association_table,back_populates='students')
#教师表
class Teacher(db.Model):
    __tablename__ = 'teacher'
    id = db.Column(db.Integer,primary_key=True,autoincrement=True)
    name = db.Column(db.String(20))
    #建立关系属性
    students = db.relationship('Student', secondary=association_table, back_populates='teachers')

关联表使用db.Table类定义,传入的第一个参数是关联表的名称。我们在关联表中定义了两个字段student_idteacher_id,这两个字段作为外键,与student.idteacher.id两个字段关联起来。

另外,这里建立的关系属性是双向的关系属性,参数secondary='关联表名称'

接着,利用关系属性添加数据

def insert():
    stu1 = Student(name='小明')
    stu2 = Student(name='小红')
    tea1 = Teacher(name='张三')
    tea2 = Teacher(name='李四')
    #学生1的老师有多个
    #由于是多对多关系,所以会采用列表的形式进行外键的orm赋值
    stu1.teachers = [tea1,tea2]
    #学生2的老师有多个
    stu2.teachers = [tea1,tea2]
    db.session.add(stu1)
    db.session.add(stu2)
    db.session.commit()

调用关系属性赋值的时候,这里需要使用列表的形式添加。

student:

teacher:

关联表:association_table:

接着,使用关系属性查询数据;

def query():
    #先查询到小明这个同学
    stu1 = Student.query.filter_by(name='小明').first()
    #使用关系属性输出对应的老师
    print(stu1.teachers)
    #反过来,查找到老师,使用关系属性输出老师对应的所有学生
    tea1 = Teacher.query.filter_by(name='张三').first()
    print(tea1.students)

输出记录

[<Teacher 1>, <Teacher 2>]

[<Student 1>, <Student 2>]

到此这篇关于Flask sqlalchemy一对多与多对一与一对一及多对多关系介绍的文章就介绍到这了,更多相关Flask sqlalchemy内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 使用sklearn之LabelEncoder将Label标准化的方法

    使用sklearn之LabelEncoder将Label标准化的方法

    今天小编就为大家分享一篇使用sklearn之LabelEncoder将Label标准化的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-07-07
  • python3解析库pyquery的深入讲解

    python3解析库pyquery的深入讲解

    做过前端开发的同志都应该知道或了解过jquery,jQuery 是一个用来处理DOM的JavaScript库。pyquery说白了就是jquery的Python版本。下面这篇文章主要给大家介绍了关于python3解析库pyquery的相关资料,需要的朋友可以参考下
    2018-06-06
  • Django如何使用jwt获取用户信息

    Django如何使用jwt获取用户信息

    这篇文章主要介绍了Django如何使用jwt获取用户信息,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下
    2020-04-04
  • Python 3.x读写csv文件中数字的方法示例

    Python 3.x读写csv文件中数字的方法示例

    在我们日常开发中经常需要对csv文件进行读写,下面这篇文章主要给大家介绍了关于Python 3.x读写csv文件中数字的相关资料,文中通过示例代码介绍的非常详细,对大家具有一定的参考学习价值,需要的朋友们下面跟着小编来一起学习学习吧。
    2017-08-08
  • 基于Python实现简易文档格式转换器

    基于Python实现简易文档格式转换器

    这篇文章主要介绍了基于Python和PyQT5实现简易的文档格式转换器,支持.txt/.xlsx/.csv格式的转换。感兴趣的小伙伴可以跟随小编一起学习一下
    2021-12-12
  • Python三目运算符(三元运算符)用法详解(含实例代码)

    Python三目运算符(三元运算符)用法详解(含实例代码)

    三元运算符在Python里被称为条件表达式,这些表达式基于真(true)/假(false)的条件判断,在Python 2.4以上才有了三元操作,下面这篇文章主要给大家介绍了关于Python三目运算符(三元运算符)用法的相关资料,需要的朋友可以参考下
    2023-02-02
  • 详解Python的整数是如何实现的

    详解Python的整数是如何实现的

    本文我们来聊一聊Python的整数,我们知道Python的整数是不会溢出的,换句话说,它可以计算无穷大的数,只要你的内存足够,它就能计算。但问题是,Python底层又是C实现的,那么它是怎么做到整数不溢出的呢?本文就来详细说说
    2022-11-11
  • python实现图像降噪

    python实现图像降噪

    这篇文章主要为大家详细介绍了python实现图像降噪,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-08-08
  • django xadmin中form_layout添加字段显示方式

    django xadmin中form_layout添加字段显示方式

    这篇文章主要介绍了django xadmin中form_layout添加字段显示方式,具有很好的 参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-03-03
  • conda查看、创建、删除、激活与退出环境命令详解

    conda查看、创建、删除、激活与退出环境命令详解

    在不同的项目中经常需要conda来配置环境,这样能够实现不同版本的python和库的随意切换,并且减少了很多不必要的麻烦,下面这篇文章主要给大家介绍了关于conda查看、创建、删除、激活与退出环境命令的相关资料,需要的朋友可以参考下
    2023-05-05

最新评论