ActiveRecord Association选择包含记录的计数

class User has_many :tickets end 

我想创建包含用户计数票据逻辑的关联,并在包含中使用它(用户has_one ticket_count)

 Users.includes(:tickets_count) 

我试过了

  has_one :tickets_count, :select => "COUNT(*) as tickets_count,tickets.user_id " ,:class_name => 'Ticket', :group => "tickets.user_id", :readonly => true User.includes(:tickets_count) ArgumentError: Unknown key: group 

在这种情况下,include中的关联查询应该使用count with group by …如何使用rails实现这一点?

更新

  • 我不能改变表结构
  • 我希望AR为包含的用户集合生成1个查询

UPDATE2

我知道SQL,我知道如何用连接选择它,但我的问题现在就像“如何获取数据”。 我的问题是关于建立我可以在包含中使用的关联。 谢谢

Update3我尝试创建像用户has_one ticket_count创建的关联,但是

  1. 看起来像has_one不支持关联扩展
  2. has_one不支持:组选项
  3. has_one不支持finder_sql

试试这个:

 class User has_one :tickets_count, :class_name => 'Ticket', :select => "user_id, tickets_count", :finder_sql => ' SELECT b.user_id, COUNT(*) tickets_count FROM tickets b WHERE b.user_id = #{id} GROUP BY b.user_id ' end 

编辑:

看起来has_one关联不支持finder_sql选项。

通过使用scope / class方法的组合,您可以轻松实现所需的function

 class User < ActiveRecord::Base def self.include_ticket_counts joins( %{ LEFT OUTER JOIN ( SELECT b.user_id, COUNT(*) tickets_count FROM tickets b GROUP BY b.user_id ) a ON a.user_id = users.id } ).select("users.*, COALESCE(a.tickets_count, 0) AS tickets_count") end end 

现在

 User.include_ticket_counts.where(:id => [1,2,3]).each do |user| p user.tickets_count end 

如果tickets表中有数百万行,则此解决方案会影响性能。 您应该考虑通过向内部查询提供WHERE来过滤JOIN结果集。

您只需使用特定用户:

 user.tickets.count 

或者,如果您希望Rails自动缓存此值。

在关联的另一端声明counter_cache => true选项

 class ticket belongs_to :user, :counter_cache => true end 

您还需要一个名为tickets_count的用户表中的列。 每次向用户添加新票证时,rails都会更新此列,因此当您对用户记录进行ftech时,只需按此列即可获得票证计数,而无需其他查询。

不漂亮,但它的工作原理:

 users = User.joins("LEFT JOIN tickets ON users.id = tickets.user_id").select("users.*, count(tickets.id) as ticket_count").group("users.id") users.first.ticket_count 

如何在执行查询的用户模型中添加方法?

您不会修改表结构,或者您也无法修改它?