按Activerecord中的特定值排序

时间:2013-12-02 22:47:26

标签: ruby-on-rails ruby activerecord rails-activerecord

在Ruby on Rails中,我正在尝试根据当前用户是否获胜来订购玩家的匹配。

排序顺序为:

  • 按当前用户是否为获胜者排序
  • 然后按created_at等排序

我无法弄清楚如何做到相同的:

Match.all.order('winner_id == ?', @current_user.id)

我知道这一行在语法上并不正确,但希望它表达的顺序必须是:

1)当前用户获胜的比赛 2)其他比赛

4 个答案:

答案 0 :(得分:6)

您可以在SQL ORDER BY子句中使用CASE表达式。但是,AR不相信在ORDER BY中使用占位符,所以你必须做这样糟糕的事情:

by_owner = Match.send(:sanitize_sql_array, [ 'case when winner_id = %d then 0 else 1 end', @current_user.id ])
Match.order(by_owner).order(:created_at)

在任何SQL数据库中都应该这样做(假设你的@current_user.id当然是整数)。

通过使用类方法作为范围,可以减少不愉快:

class Match < ActiveRecord::Base
  def self.this_person_first(id)
    by_owner = sanitize_sql_array([ 'case when winner_id = %d then 0 else 1 end', id])
    order(by_owner)
  end
end

# and later...
Match.this_person_first(@current_user.id).order(:created_at)

隐藏肮脏。

答案 1 :(得分:1)

这可以使用 Arel 实现,而无需编写任何原始 SQL!

matches = Match.arel_table
Match
  .order(matches[:winner_id].eq(@current_user.id).desc)
  .order(created_at: :desc)

适用于 Postgres 12 / Rails 6.0.3,没有任何安全警告 demo

答案 2 :(得分:0)

如果你想在ruby方面(而不是SQL方面)进行排序,那么你可以使用Array#sort_by方法:

query.sort_by(|a| a.winner_id == @current_user.id)

如果你正在处理更大的查询,那么你应该坚持SQL的一面。

答案 3 :(得分:-1)

我会构建一个查询,然后在构建之后执行它(主要是因为你可能没有@current_user。所以,像这样:

query = Match.scoped

query = query.order("winner_id == ?", @current_user.id) if @current_user.present?

query = query.order("created_at")

@results = query.all