16

例子

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 和 group by ...我如何使用 rails 来实现这个?

更新

  • 我无法更改表结构
  • 我希望 AR 为包含包含的用户集合生成 1 个查询

更新2

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

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

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

4 回答 4

16

试试这个:

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方法的组合轻松实现您想要的

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 结果集。

于 2012-02-21T03:14:32.360 回答
10

您可以简单地用于特定用户:

user.tickets.count

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

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

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

您还需要在您的用户表中有一列名为tickets_count。每次向用户添加新票时,rails 都会更新此列,因此当您对用户记录进行 ftech 时,您可以简单地访问此列来获取票数,而无需额外查询。

于 2012-02-09T11:04:12.963 回答
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
于 2012-02-13T21:23:28.393 回答
0

在执行查询的用户模型中添加一个方法怎么样?

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

于 2012-02-19T22:32:28.527 回答