1

我有一个模型 Laps,它属于:Car

所以每辆车都有很多圈,但我需要做一个查询,我在哪里拉进前 10 最快圈,但每辆车只拉最快圈。换句话说,我不希望一辆车被允许在列表中的前 10 最快圈中拥有 5 圈。

每圈都有一个 :car_id 字段。我想在此列上分组,然后从分组中拉出带有 min(Lap.time) 行的圈。(也就是那个独特的:car_id 中最快的单圈时间)。

那么对于 Postgres,我最好的方法是什么?我可以先订购所有圈,然后分组,然后从组中拉出第一圈吗?或者分组不会保持排序顺序?

这是我的 Lap 模型架构:

  create_table "laps", force: true do |t|
    t.decimal  "time"
    t.string   "video_url"
    t.integer  "car_id"
    t.datetime "created_at"
    t.datetime "updated_at"
    t.integer  "user_id"
    t.boolean  "approved"
  end

我必须为此使用两个组合查询吗?

通过这样做,我可以获得一个独特的汽车 ID 圈:

select('DISTINCT ON (car_id) *')

但是,我需要订购圈数,以便它获得每个 car_id 的 min(lap.time)。所以当我发出这样的订单时:

select('DISTINCT ON (car_id) *').order('car_id, time ASC').sort_by! {|ts| ts.time}

这可行,但似乎是一种奇怪的方法。一旦我尝试更改订单,例如从订单中删除 car_id,我就会收到 postgres 错误。

4

2 回答 2

4

正如您所发现的,DISTINCT ON 在这里对您不起作用,因为它与您想要排序的第一个术语(时间)不匹配。您需要使用 GROUP BY:

Lap.group(:car_id).limit(10).minimum(:time)

或者,您可以创建一个窗口子查询 - 但构建起来相当麻烦。如果您暂时需要实际的单圈信息,您可能必须走这条路线:

subquery = Lap.select('lap.*, ROW_NUMBER() OVER ( PARTITION BY car_id ORDER BY time, car_id ) as rowNum').to_sql
Lap.scoped.from(Arel.sql("(#{subquery}) fast_laps"))
   .where('rowNum = 1')
   .order('time, car_id')
   .limit(10)
于 2013-09-03T16:35:20.837 回答
0

这就是我所做的及其工作。如果有更好的方法来解决这些问题,请发布您的答案:

在我的模型中:

def self.fastest_per_car
    select('DISTINCT ON (car_id) *').order('car_id, time ASC').sort_by! {|ts| ts.time}
end
于 2013-09-03T16:19:29.813 回答