首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >Rails使用相关表中的数据进行查找和排序

Rails使用相关表中的数据进行查找和排序
EN

Stack Overflow用户
提问于 2009-12-22 08:10:48
回答 2查看 3.3K关注 0票数 3

我关联了两个模型BusinessesRatings。一个业务可以进行多次评级,每次评级都会存储一个从0到5的整数。

我想创建一个“名人堂”页面,在那里我列出了排名前10的企业,根据平均评级对企业进行排序(这将使用Ratings模型)并将结果限制在10个。

我不确定如何为控制器编写hall_of_fame方法,或者我是否也需要一个帮助器来实现这一点?

EN

回答 2

Stack Overflow用户

回答已采纳

发布于 2009-12-22 08:46:27

假设您在两个模型之间具有has_many和belongs_to关系,您可以尝试使用(非常方便的)Rails支持的MySQL GROUP BY子句:

代码语言:javascript
复制
@hall_of_fame = Business.find(
                               :all, 
                               :joins => :ratings, 
                               :group => 'business_id', 
                               :order => 'AVG(ratings.rating) DESC', 
                               :limit => 10
                             )

如果要添加平均评级,可以将其包括在:select参数中,如下所示:

代码语言:javascript
复制
@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参数中选择。

您可能希望在模型中使用此代码创建方法,而不是将其放在控制器中,但这取决于您。

票数 7
EN

Stack Overflow用户

发布于 2009-12-22 08:49:14

在你的控制器中,你可以抓取排名前十的业务,类似于:

代码语言:javascript
复制
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'。因此,它将返回一个有序的散列。当你迭代它的时候,你得到了你想要的东西。

也许更好的方法是向您的业务模型添加一个类方法,如下所示:

代码语言:javascript
复制
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

然后在控制器中,你可以这样做:

代码语言:javascript
复制
top10 = Business.getTop10
票数 0
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/1943449

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档