5

我对 SQLAlchemy 和别名有疑问。我正在尝试对一个名为 Task 的表进行自联接,但 SQLAlchemy 将我的别名表与非别名表混淆了。

我的查询要复杂得多,但这是一个简化版本:

baseTask = aliased(model.Task)

query = model.session.query(model.Task.name)\
       .select_from(baseTask)\
       .join((model.Task, model.Task.taskid==baseTask.taskid))

The resulting query looks like this:    
SELECT task_1.name 
FROM task AS task_1 INNER JOIN task ON task_1.taskid = task_1.taskid

正如您所看到的,它在选择和连接条件中都将未别名表与别名 task_1 混淆了。

通过在两个表上使用别名,它可以工作:

baseTask = aliased(model.Task)
task = aliased(model.Task)

query = model.session.query(task.name)\
        .select_from(baseTask)\
        .join((task, task.taskid==baseTask.taskid))

This gives the desired result:
SELECT task_1.name 
FROM task AS task_2 INNER JOIN task AS task_1 ON task_1.taskid = task_2.taskid

当我在两个表上使用别名时,它不会混淆它们,并且结果查询正是我想要的。问题是由于各种原因我不能在我加入的表上使用别名,这是由于动态生成查询的应用程序设计。

我正在使用 SQLAlchemy 0.6.8,我尝试升级到 0.7.9,但我仍然遇到这个问题。关于如何解决这个问题的任何想法?

4

1 回答 1

2

不幸的是,select_from() 现在所做的两件特定事情之间有一些功能重叠,其中一件是你可以说query(Task).select_from(<some SELECT statement>),它会根据 重新定位“任务”,<some SELECT statement>另一件是你可以说它首先从哪里加入。

添加了http://www.sqlalchemy.org/trac/ticket/2635,也许在 0.8 中我们可以让它在这种特定情况下做出更好的决策。

更新:感谢您对此的报告,我过去在自己的工作中也遇到过,但没能深入了解,这很简单。如果您现在想使用 0.8 提示,您的用例应该可以工作。

这是修复的一些背景:http ://docs.sqlalchemy.org/en/latest/changelog/changelog_08.html#change-8d958f76ee47f715536​​5772401087d1f

潜在的 0.6.8 解决方法

query = session.query(Task.name)

query._from_obj += (baseTask._AliasedClass__alias, )

query = query.join((Task, Task.taskid == baseTask.taskid))
于 2012-12-13T23:18:08.370 回答