优化查询PostgreSql-8.4

时间:2011-04-20 04:56:34

标签: ruby-on-rails-3 postgresql

我的轨道控制器编码如下:

@checked_contact_ids = @list.contacts.all(
  :conditions => {
    "contacts_lists.contact_id" => @list.contacts.map(&:id),
    "contacts_lists.is_checked" => true
  }
).map(&:id)

它相当于sql

SELECT *
FROM "contacts"
INNER JOIN "contacts_lists" ON "contacts".id = "contacts_lists".contact_id
WHERE ("contacts_lists".list_id = 67494 ) 

上面的查询需要更多的时间来运行,我想用另一种方式以最短的时间运行相同的查询。

有谁知道请注意我或者可能吗?或上述查询是否足以给出输出?

我正在等待信息...................

1 个答案:

答案 0 :(得分:1)

我认为原始AR查询的主要问题是它根本没有进行任何连接;你通过@list.contacts将一堆对象从数据库中拉出来,然后将大部分工作扔掉以获得ID。

第一步是将"contacts_lists.contact_id" => @list.contacts.map(&:id)替换为:joins => 'contact_lists',但你仍然会从数据库中取出一堆东西,实例化一堆对象,然后全部抛出远离.map(&:id)获取身份证号码。

你已经知道了SQL,所以我可能会通过List模型上的便捷方法(或者@list)直接使用SQL,如下所示:

def checked_contact_ids
    connection.execute(%Q{
        SELECT contacts.id
        FROM contacts
        INNER JOIN contacts_lists ON contacts.id = contacts_lists.contact_id
        WHERE contacts_lists.list_id    = #{self.id}
          AND contacts_lists.is_checked = 't'
    }).map { |r| r['id'] }
end

然后,在你的控制器中:

@checked_contact_ids = @list.checked_contact_ids

如果速度不够快,请查看contacts_lists表上的索引。

当你确切地知道你需要什么数据并且你需要它时,没有充分的理由不直接使用SQL;只需将SQL隔离在模型中,就不应该有任何问题。