0

0

SQLAlchemy 如何获取子类对象?

聖光之護

聖光之護

发布时间:2025-09-13 20:45:00

|

509人浏览过

|

来源于php中文网

原创

sqlalchemy 如何获取子类对象?

第一段引用上面的摘要:

本文档旨在解决 SQLAlchemy 中关系映射后,父类对象无法立即访问到已关联子类对象的问题。通过示例代码,详细解释了 SQLAlchemy 中关系建立的时机,以及如何通过 flush 操作或手动关联来正确获取关联的子类对象。同时,提供了两种测试用例,帮助读者理解和掌握 SQLAlchemy 中关系操作的细节。

在 SQLAlchemy 中,使用 relationship 定义父类和子类之间的关系是一种常见的做法。然而,新手在使用时可能会遇到一个问题:在将父类和子类对象添加到 Session 后,父类对象的 children 属性并没有立即更新,仍然是一个空列表。本文将深入探讨这个问题,并提供解决方案。

理解 SQLAlchemy 的关系建立时机

SQLAlchemy 默认情况下,并不会在对象添加到 Session 后立即解析关系。关系的建立通常发生在 flush 或 commit 操作之后。这是因为 SQLAlchemy 需要等待事务提交,才能确保数据库中的数据一致性。

示例代码:

以下代码演示了在 flush 操作前,parent.children 属性为空的情况。

DreamGen
DreamGen

一个AI驱动的角色扮演和故事写作的平台

下载
from sqlalchemy.orm import declarative_base, relationship, Session
from sqlalchemy import Column, String, Integer, ForeignKey, create_engine

Base = declarative_base()

class Parent(Base):
    __tablename__ = 'parents'

    id = Column(Integer, primary_key=True)
    name = Column(String(20))

    children = relationship('Child', back_populates='parent')


class Child(Base):
    __tablename__ = 'children'

    id = Column(Integer, primary_key=True)
    parent_id = Column(Integer, ForeignKey('parents.id'))
    name = Column(String(20))

    parent = relationship('Parent', back_populates='children')

# Replace with your actual database connection string
engine = create_engine('sqlite:///:memory:')
Base.metadata.create_all(engine)

with Session(engine) as session:
    mother = Parent(id=1, name='Sarah')
    c1 = Child(id=22, parent_id=mother.id, name='Alice')
    c2 = Child(id=23, parent_id=mother.id, name='Bob')

    session.add(mother)
    session.add(c1)
    session.add(c2)

    print(mother.children)  # 输出: []

    session.flush()

    print(mother.children)  # 输出: [<__main__.Child object at ...>, <__main__.Child object at ...>]

在上面的代码中,mother.children 在 session.flush() 之前输出的是空列表。flush() 操作将对象的状态同步到数据库,并解析了对象之间的关系。flush后,mother.children包含了 c1 和 c2 对象。

解决方案:手动关联对象

除了等待 flush 操作之外,也可以手动关联对象,从而立即访问到子类对象。

示例代码:

from sqlalchemy.orm import declarative_base, relationship, Session
from sqlalchemy import Column, String, Integer, ForeignKey, create_engine

Base = declarative_base()

class Parent(Base):
    __tablename__ = 'parents'

    id = Column(Integer, primary_key=True)
    name = Column(String(20))

    children = relationship('Child', back_populates='parent')


class Child(Base):
    __tablename__ = 'children'

    id = Column(Integer, primary_key=True)
    parent_id = Column(Integer, ForeignKey('parents.id'))
    name = Column(String(20))

    parent = relationship('Parent', back_populates='children')

# Replace with your actual database connection string
engine = create_engine('sqlite:///:memory:')
Base.metadata.create_all(engine)


with Session(engine) as session:
    c1 = Child(id=22, name='Alice')
    c2 = Child(id=23, name='Bob')

    mother = Parent(id=1, name='Sarah', children=[c1, c2])

    session.add(mother)
    session.add(c1)
    session.add(c2)

    print(mother.children) # 输出: [<__main__.Child object at ...>, <__main__.Child object at ...>]

    session.flush()

在这个例子中,我们在创建 mother 对象时,直接将 c1 和 c2 对象添加到 children 列表中。这样,在添加到 Session 之前,mother.children 就已经包含了子类对象。需要注意的是,即使手动关联了对象,仍然需要执行 flush 操作,才能将对象的 parent_id 更新到数据库中。

注意事项和总结

  • 理解 SQLAlchemy 关系建立的时机非常重要。默认情况下,关系在 flush 或 commit 操作后才会建立。
  • 可以使用 flush 操作来强制 SQLAlchemy 解析关系。
  • 可以手动关联对象,从而立即访问到子类对象。
  • 即使手动关联了对象,也需要执行 flush 操作,以确保数据库中的数据一致性。

通过本文的学习,相信你已经掌握了 SQLAlchemy 中获取子类对象的方法。在实际开发中,根据具体需求选择合适的方式,可以更高效地操作数据库。

相关专题

更多
session失效的原因
session失效的原因

session失效的原因有会话超时、会话数量限制、会话完整性检查、服务器重启、浏览器或设备问题等等。详细介绍:1、会话超时:服务器为Session设置了一个默认的超时时间,当用户在一段时间内没有与服务器交互时,Session将自动失效;2、会话数量限制:服务器为每个用户的Session数量设置了一个限制,当用户创建的Session数量超过这个限制时,最新的会覆盖最早的等等。

302

2023.10.17

session失效解决方法
session失效解决方法

session失效通常是由于 session 的生存时间过期或者服务器关闭导致的。其解决办法:1、延长session的生存时间;2、使用持久化存储;3、使用cookie;4、异步更新session;5、使用会话管理中间件。

706

2023.10.18

cookie与session的区别
cookie与session的区别

本专题整合了cookie与session的区别和使用方法等相关内容,阅读专题下面的文章了解更详细的内容。

88

2025.08.19

数据库三范式
数据库三范式

数据库三范式是一种设计规范,用于规范化关系型数据库中的数据结构,它通过消除冗余数据、提高数据库性能和数据一致性,提供了一种有效的数据库设计方法。本专题提供数据库三范式相关的文章、下载和课程。

334

2023.06.29

如何删除数据库
如何删除数据库

删除数据库是指在MySQL中完全移除一个数据库及其所包含的所有数据和结构,作用包括:1、释放存储空间;2、确保数据的安全性;3、提高数据库的整体性能,加速查询和操作的执行速度。尽管删除数据库具有一些好处,但在执行任何删除操作之前,务必谨慎操作,并备份重要的数据。删除数据库将永久性地删除所有相关数据和结构,无法回滚。

2068

2023.08.14

vb怎么连接数据库
vb怎么连接数据库

在VB中,连接数据库通常使用ADO(ActiveX 数据对象)或 DAO(Data Access Objects)这两个技术来实现:1、引入ADO库;2、创建ADO连接对象;3、配置连接字符串;4、打开连接;5、执行SQL语句;6、处理查询结果;7、关闭连接即可。

346

2023.08.31

MySQL恢复数据库
MySQL恢复数据库

MySQL恢复数据库的方法有使用物理备份恢复、使用逻辑备份恢复、使用二进制日志恢复和使用数据库复制进行恢复等。本专题为大家提供MySQL数据库相关的文章、下载、课程内容,供大家免费下载体验。

251

2023.09.05

vb中怎么连接access数据库
vb中怎么连接access数据库

vb中连接access数据库的步骤包括引用必要的命名空间、创建连接字符串、创建连接对象、打开连接、执行SQL语句和关闭连接。本专题为大家提供连接access数据库相关的文章、下载、课程内容,供大家免费下载体验。

319

2023.10.09

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

74

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Laravel 5.8 中文文档手册
Laravel 5.8 中文文档手册

共74课时 | 82.2万人学习

SESSION实现登录与验证
SESSION实现登录与验证

共10课时 | 9.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号