我一直在优化Rails应用程序,但我遇到了一个特殊的查询:
def self.random_selection(n)
items = scoped(:joins => "JOIN (SELECT id
FROM #{table_name}
WHERE medias_count > 0
ORDER BY RAND()
LIMIT #{n.to_i}
) AS random_ids
ON #{table_name}.id = random_ids.id"
)
items.each do |genre|
genre.medias.sort! do |x,y|
y.vote_total <=> x.vote_total
end
end
items
end
这个想法是,它将随机选择一些有媒体的流派。一旦被选中,它将在评分最高的媒体上进行排序,我认为选择"顶级媒体"并在视图中使用它。
这是一个相当昂贵、丑陋的查询,我想了解一些优化它的方法
我可以将所选媒体滚动到原始查询中吗?
我应该从另一个方向来处理这个问题,随机选择高收视率的媒体,并从中获取类型吗?(也可以接受,但如果它没有提供任何改进,那么他们就没有意义了)
我在InnoDB中使用Rails 3、Ruby 1.9.2和MySQL。
我的解决方案
class Genre
scope :having_media, where('medias_count > 0')
scope :random, lambda { |limit| where(:id => random_ids(limit)) }
def self.random_ids(limit)
having_media.select('id').sample(limit).map(&:id)
end
def self.random_selection(limit)
random(10).includes(:medias).each do |genre|
genre.medias.sort! do |x,y|
y.vote_total <=> x.vote_total
end
end
end
end
class Media
scope :voted_higher, reorder('vote_total desc')
end
@random_genres = Genre.random_selection(10)