我正在尝试使用嵌套的select子句实现一个select,以查找没有任何子节点的父节点。我的表格(从根本上简化)如下:
class Person(Base):
__tablename__ = "person"
id = Column(Integer, primary_key=True)
name = Column(String, nullable=True, default=None)
class ChildTable(Base):
__tablename__ = "foo"
id = Column(Integer, primary_key=True)
data = Column(String)
person_id = Column(Integer, ForeignKey("person.id"), nullable=True)
在sql中,我的查询将是
select id from person where id not in (select person_id from foo);
如何在Python和sqlalchemy中实现这一点?我发现了这个sqlalchemy, select using reverse-inclusive (not in) list of child column values,但它依赖于声明的关系。我没有使用关系,因为我遇到了很多问题,主要是其他线程过期或更改其他线程缓存的数据,以及foo.person_id可以为null而不引用任何内容。
我该怎么做?我尝试了in_和任何运算符的组合,抛出了不同的异常。例如,这次尝试:
empty_persons = config.Session.query(Person).filter(Person.id.notin_(ChildTable.\
person_id)).all()
失败:in_()接受表达式列表或可选表达式。
这可能是一件简单的事情,但我不明白应该怎么做。
[R
答案 0 :(得分:12)
使用子查询:
sub_stmt = config.Session.query(ChildTable.person_id)
stmt = config.Session.query(Person).filter(~Person.id.in_(sub_stmt))
empty_persons = stmt.all()
发出以下sql:
SELECT person.id AS person_id, person.name AS person_name
FROM person
WHERE person.id NOT IN (SELECT foo.person_id AS foo_person_id
FROM foo)
使用联接:
stmt = config.Session.query(Person).outerjoin(ChildTable).filter(ChildTable.person_id.is_(None))
empty_persons = stmt.all()
发出以下sql:
SELECT person.id AS person_id, person.name AS person_name
FROM person LEFT OUTER JOIN foo ON person.id = foo.person_id
WHERE foo.person_id IS NULL
我认为两者都能实现您想要的结果集。