实施例
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实现此功能?
更新
UPDATE2
我知道SQL,我知道如何用连接选择它,但我的问题现在就像“如何获取数据”。我的问题是关于建立我可以在包含中使用的关联。感谢
UPDATE3 我尝试创建像用户has_one ticket_count创建的关联,但
答案 0 :(得分:13)
试试这个:
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结果集。
答案 1 :(得分:9)
您只需使用特定用户:
user.tickets.count
或者如果您希望Rails自动缓存此值。
声明counter_cache =&gt;关联另一边的真实选项
class ticket
belongs_to :user, :counter_cache => true
end
您还需要一个名为tickets_count的用户表中的列。 这样,每次向用户添加新票据时,rails都会更新此列,因此当您修改用户记录时,您只需按此列即可获得票证计数而无需其他查询。
答案 2 :(得分: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
答案 3 :(得分:0)
如何在执行查询的用户模型中添加方法?
您不会修改表结构,或者您也无法修改它?