SQLAlchemy多对多动态延迟加载不返回AppenderQuery

时间:2020-08-07 09:13:23

标签: python orm sqlalchemy flask-sqlalchemy relationship

我正在尝试掌握SQLAlchemy ORM的backref延迟加载动态功能。

我有3个表和2个链接表。

course_members = Table('course_members', Base.metadata,
            Column('user_id', Integer, ForeignKey('users.id')),
            Column('course_id', Integer, ForeignKey('courses.id'))
            )

course_roles = Table('course_roles', Base.metadata,
              Column('role_id', Integer, ForeignKey('roles.id')),
              Column('course_id', Integer, ForeignKey('courses.id'))
              )

class User(Base):
    id = Column(Integer, primary_key=True, autoincrement=True)
    name = Column(String(200), nullable=False)

class Course(Base, Jsonify):
    id = Column(Integer, primary_key=True, autoincrement=True)
    members = relationship('User', secondary=course_members, backref=backref('courses', lazy='dynamic'))
    roles = relationship('Role', secondary=course_roles, backref=backref('roles', lazy='dynamic'))

class Role(Base):

    id = Column(Integer, primary_key=True, autoincrement=True)
    role_id = Column(Integer, nullable=False)
    user_id = Column(Integer, ForeignKey('users.id'))
    user = relationship('User', backref=backref("roles", lazy='dynamic'))
    
    UniqueConstraint('role_id', 'user_id', name='role_combination')

通过用户查询:

print(type(user.roles))
print(user.roles.filter_by(id=1).first())
# print(user.courses.first() <-- Works fine too.

<class 'sqlalchemy.orm.dynamic.AppenderQuery'>
{id: 1, 'user_id': 1, role_id: 3}

但有课程查询:

print(type(course.roles))
print(course.roles.filter_by(id=1).first())

<class 'sqlalchemy.orm.collections.InstrumentedList'>
AttributeError: 'InstrumentedList' object has no attribute 'filter_by'

与会员尝试时结果相同。

将成员和课程用作工作的列表对象:

course.roles[0].user.first()

但是我的课程查询确实错过了AppenderQuery类的功能。

这是正常行为还是我缺少什么?

1 个答案:

答案 0 :(得分:0)

在撰写问题并进行研究时,我在这篇add dynamic to the other side帖子中找到了问题的答案。

看到答案后,我也了解了更多的关系功能。

members = relationship('User', secondary=course_members, lazy='dynamic', backref=backref('courses', lazy='dynamic'))
roles = relationship('Role', secondary=course_roles, lazy='dynamic', backref=backref('roles', lazy='dynamic'))

成员和角色关系是父类的功能,而backref = *则是子类的关系功能。我花了更多的时间来实现这一目标。但是,通过在关系函数中都使用lazy ='dynamic'作为参数,针对backref参数的backref函数将其有效地应用于关系的两端。

现在,用户,课程和角色将作为AppenderQuery返回。

希望这将有助于其他人寻找问题。