ICode9

精准搜索请尝试: 精确搜索
首页 > 数据库> 文章详细

python – sqlalchemy:关闭声明性多态连接?

2019-07-05 03:58:03  阅读:274  来源: 互联网

标签:python sqlalchemy declarative


sqlalchemy中是否有一种方法可以在单个查询中关闭声明性的多态连接加载?大部分时间都很好,但我有:

class A(Base) : 
   discriminator = Column('type', mysql.INTEGER(1), index=True, nullable=False)
   __mapper_args__ = { 'polymorphic_on' : discriminator }
   id = Column(Integer, primary_key=True)
   p = Column(Integer)

class B(A) : 
   __mapper_args__ = { 'polymorphic_identity' : 0 }
   id = Column(Integer, primary_key=True)
   x = Column(Integer)

class C(A) : 
   __mapper_args__ = { 'polymorphic_identity' : 1 }
   id = Column(Integer, primary_key=True)
   y = Column(String)

我想进行一个查询,以便得到所有A.ids,其中B.x> 10,如果A实际上是B,或者C.y ==’blah’,如果A实际上是C,则全部按p排序.

为了迭代地做,我只是从第一部分开始 – “如果A实际上是B,则获得所有A.id,其中B.x> 10.”所以我想我会从外部联接开始:

session.query(A.id).outerjoin((B, B.id == A.id)).filter(B.x > 10)

…除了似乎没有办法避免让外连接((B,B.id == A.id))子句生成A中所有内容与子选择中B中所有内容的完全连接.如果B没有从A继承,那么这种情况就不会发生,所以我认为这是多态声明代码生成.有没有办法把它关掉?或者强迫外部连接做我想做的事情?

我想要的是这样的:

select a.id from A a left outer join B b on b.id == a.id where b.x > 10

但取而代之的是:

select a.id from A a left outer join (select B.id, B.x, A.id from B inner join A on B.id == A.id)

……顺便说一句,如果不可能,那么后者效率会低于前者吗? sql引擎是否会实际执行内连接,还是会忽略它?

解决方法:

您可以尝试单独为每个子类构建查询,然后将它们组合在一起.在查询B.id时,SQLAlchemy隐式加入超类并返回A.id,因此对B.id和C.id选择联合仅返回单个列.

>>> b_query = session.query(B.id).filter(B.x > 10)
>>> c_query = session.query(C.id).filter(C.y == 'foo')
>>> print b_query.union(c_query)
SELECT anon_1."A_id" AS "anon_1_A_id" 
FROM (SELECT "A".id AS "A_id" 
FROM "A" JOIN "B" ON "A".id = "B".id 
WHERE "B".x > ? UNION SELECT "A".id AS "A_id" 
FROM "A" JOIN "C" ON "A".id = "C".id 
WHERE "C".y = ?) AS anon_1

您仍然可以获得一个子选择,但只有一个“连接层” – 外部选择只是重命名该列.

标签:python,sqlalchemy,declarative
来源: https://codeday.me/bug/20190705/1383686.html

本站声明: 1. iCode9 技术分享网(下文简称本站)提供的所有内容,仅供技术学习、探讨和分享;
2. 关于本站的所有留言、评论、转载及引用,纯属内容发起人的个人观点,与本站观点和立场无关;
3. 关于本站的所有言论和文字,纯属内容发起人的个人观点,与本站观点和立场无关;
4. 本站文章均是网友提供,不完全保证技术分享内容的完整性、准确性、时效性、风险性和版权归属;如您发现该文章侵犯了您的权益,可联系我们第一时间进行删除;
5. 本站为非盈利性的个人网站,所有内容不会用来进行牟利,也不会利用任何形式的广告来间接获益,纯粹是为了广大技术爱好者提供技术内容和技术思想的分享性交流网站。

专注分享技术,共同学习,共同进步。侵权联系[81616952@qq.com]

Copyright (C)ICode9.com, All Rights Reserved.

ICode9版权所有