ActiveRecord Association选择包含记录的计数

Fiv*_*ell 15 ruby activerecord ruby-on-rails active-relation ruby-on-rails-3

class User  
  has_many :tickets 
end
Run Code Online (Sandbox Code Playgroud)

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

Users.includes(:tickets_count)
Run Code Online (Sandbox Code Playgroud)

我试过了

  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
Run Code Online (Sandbox Code Playgroud)

在这种情况下,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

Har*_*tty 14

试试这个:

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
Run Code Online (Sandbox Code Playgroud)

编辑:

看起来has_one协会不支持该finder_sql选项.

您可以使用scope/ classmethods 组合轻松实现所需

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
Run Code Online (Sandbox Code Playgroud)

现在

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

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

  • 非常好的解决方案,以避免令人作呕的黑客`:counter_cache`.只有改变我必须做的(我正在使用Postgres)正在改变`.select("users.*,a.tickets_count")``到`.select("users.*,COALESCE(a.tickets_count,0)AS tickets_count ")` (2认同)

小智 9

您只需使用特定用户:

user.tickets.count
Run Code Online (Sandbox Code Playgroud)

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

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

class ticket
  belongs_to :user, :counter_cache => true
end
Run Code Online (Sandbox Code Playgroud)

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


Thi*_*ilo 7

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

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
Run Code Online (Sandbox Code Playgroud)