我关联了两个模型Businesses和Ratings。一个业务可以进行多次评级,每次评级都会存储一个从0到5的整数。
我想创建一个“名人堂”页面,在那里我列出了排名前10的企业,根据平均评级对企业进行排序(这将使用Ratings模型)并将结果限制在10个。
我不确定如何为控制器编写hall_of_fame方法,或者我是否也需要一个帮助器来实现这一点?
发布于 2009-12-22 08:46:27
假设您在两个模型之间具有has_many和belongs_to关系,您可以尝试使用(非常方便的)Rails支持的MySQL GROUP BY子句:
@hall_of_fame = Business.find(
:all,
:joins => :ratings,
:group => 'business_id',
:order => 'AVG(ratings.rating) DESC',
:limit => 10
)如果要添加平均评级,可以将其包括在:select参数中,如下所示:
@hall_of_fame = Business.find(
:all,
:select => 'businesses.name, AVG(ratings.rating)'
:joins => :ratings,
:group => 'business_id',
:order => 'AVG(ratings.rating) DESC',
:limit => 10
)当然,如果表之间没有冲突的列名,您就可以安全地删除主要的“业务”。还有“收视率”从:select和:order参数中选择。
您可能希望在模型中使用此代码创建方法,而不是将其放在控制器中,但这取决于您。
发布于 2009-12-22 08:49:14
在你的控制器中,你可以抓取排名前十的业务,类似于:
top10 = Business.average('ratings.rating', :joins => 'INNER JOIN ratings ON businesses.id = ratings.business_id', :group => 'businesses.id', :order => 'avg_ratings_rating DESC', :limit => 10 )
# example of using results of above line
top10.each { |id, avg| puts Business.find(id).name + " score: " + avg.to_s }您可以编写:order => 'avg(ratings.rating)'而不是:order => 'avg_ratings_rating'。因此,它将返回一个有序的散列。当你迭代它的时候,你得到了你想要的东西。
也许更好的方法是向您的业务模型添加一个类方法,如下所示:
def self.getTop10
average('ratings.rating', :joins => 'INNER JOIN ratings ON businesses.id = ratings.business_id', :group => 'businesses.id', :order => 'avg_ratings_rating DESC', :limit => 10 )
end然后在控制器中,你可以这样做:
top10 = Business.getTop10https://stackoverflow.com/questions/1943449
复制相似问题