SQLAlchemy查询过滤器子属性

14

我的模型由父级和子级组成,两者之间是一对一的关系:

class Parent(Base):
    __tablename__ = 'parent'
    id = Column(Integer, primary_key=True)
    name = Column(String)
    child = relationship("Child", backref="parent", uselist=False, lazy='joined')


class Child(Base):
    __tablename__ = 'child'
    child_id = Column(Integer, ForeignKey(Parent.id), primary_key=True)
    value = Column(Integer)

我的测试数据如下:

q = s.query(Parent)
pd.read_sql(q.statement,s.bind) 
    id  name  child_id  value
    1      a         1     10
    2      b         2     20
    3      c         3     30

现在我想使用这个查询仅获取具有child.value > 20 的父级:

q = s.query(Parent).filter(Parent.child.value > 20)

但是出现了这个错误:

AttributeError: Neither 'InstrumentedAttribute' object nor 'Comparator' object 
associated with Parent.child has an attribute 'value'

当然,我可以直接查询子类,但我的目标是检索父对象。

2个回答

25

你应该修改你的查询。

# version-1: use JOIN
q = s.query(Parent).join(Child, Parent.child).filter(Child.value > 20)

# or:
# version-2: use EXISTS
q = s.query(Parent).filter(Parent.child.has(Child.value > 20))

谢谢@van,但我正在尝试复制Quantopian API的get_fundamentals功能,也可以参考这个问题:http://stackoverflow.com/questions/40497528/sqlalchemy-model-for-quantopian-get-fundamentals。我无法理解他们如何定义基础模型。 - kostia
我理解你的问题是“如何获取Parent对象?” ,我相信我的回答已经解决了这个问题。至于你其他的问题以及它与Quantopian的链接:问题不够清晰 - 哪些工作?哪些不工作?我只能确认,使用多级属性导航(例如 Parent.child[.subchild].property)来访问关系在SQLAlchemy中不起作用。Quantopian的示例也没有显示出相反的情况 - fundamentals 是模块,而不是映射类。 - van
是的,@van,你说得完全正确。我只是想指向我的另一个问题,这个问题源于那个问题,你给了我一个很好的提示。现在我也认为基础知识必须是模块。 - kostia

1

我知道你提到了不想查询子类,但实际上这是在幕后发生的(SQLAlchemy只是将其隐藏起来),因此你可以查询。然后,你可以通过backref简单地访问父对象。由于你指定了lazy=joined,速度将完全相同。

q = s.query(Child).filter(Child.value > 20)
parent_obj = q.parent

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接