我正在尝试掌握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类的功能。
这是正常行为还是我缺少什么?
答案 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返回。
希望这将有助于其他人寻找问题。