如何在不使用包含的情况下避免使用n + 1?

时间:2017-04-28 21:42:27

标签: ruby-on-rails postgresql ruby-on-rails-4 join

我有两个型号

class Task < AR::Base
  habtm :user_groups
end

class UserGroup < AR::Base
  habtm :tasks
end

要求:

t = Task.where(id: 45734)
t.user_group_ids #[54, 523, 78]

以上行会导致2次查询。

我希望在不使用includes的情况下在一个查询中执行此操作。

我的尝试:

tasks = Task.joins(:user_groups).select('tasks.id, user_groups.id as ug_id').where(customer_id: 5)

但即使对于有多个组关联的任务,它也只检索单个用户组。

更新

Max建议:

t = Task.joins('LEFT OUTER JOIN tasks_user_groups ON tasks_user_groups . analysis_task_id  =  tasks . id  INNER JOIN  user_groups  ON  user_groups . id  =  tasks_user_groups . user_group_id').select('user_groups.id').where(id: 36299)

现在我得到两个具有相同ID但有两个不同用户组的任务。

请建议如何实现它。

1 个答案:

答案 0 :(得分:0)

.joins使用INNER LEFT JOIN,因此它只会为users_groups中匹配的任务提供行。

.includes通过使用两个单独的查询解决了这个问题。

您想要的是外连接。

# Rails 5
Task.left_outer_joins(:user_groups)

# Rails 4
tasks = Task.joins('LEFT OUTER JOIN user_groups ON user_groups.task_id = tasks.id')

请参阅:

What is the difference between "INNER JOIN" and "OUTER JOIN"?