Запрос на ранжирование строк в группах

Я использую Apache Derby 10.10.

У меня есть список участников, и я хотел бы рассчитать их рейтинг в своей стране, например так:

|        Country |         Participant | Points | country_rank |
|----------------|---------------------|--------|--------------|
|      Australia |      Bridget Ciriac |      1 |            1 |
|      Australia |     Austin Bjorklun |      4 |            2 |
|      Australia |        Carrol Motto |      7 |            3 |
|      Australia |     Valeria Seligma |      8 |            4 |
|      Australia |     Desmond Miyamot |     27 |            5 |
|      Australia |      Maryjane Digma |     33 |            6 |
|      Australia |       Kena Elmendor |     38 |            7 |
|      Australia |         Emmie Hicke |     39 |            8 |
|      Australia |        Kaitlyn Mund |     50 |            9 |
|      Australia |    Alisia Vitaglian |     65 |           10 |
|      Australia |          Anika Bulo |     65 |           11 |
|             UK |          Angle Ifil |      2 |            1 |
|             UK |     Demetrius Buelo |     12 |            2 |
|             UK |      Ermelinda Mell |     12 |            3 |
|             UK |         Adeline Pee |     21 |            4 |
|             UK |     Alvera Cangelos |     23 |            5 |
|             UK |   Keshia Mccalliste |     23 |            6 |
|             UK |        Alayna Rashi |     24 |            7 |
|             UK |    Malinda Mcfarlan |     25 |            8 |
|  United States |     Gricelda Quirog |      3 |            1 |
|  United States |      Carmina Britto |      5 |            2 |
|  United States |         Noemi Blase |      6 |            3 |
|  United States |        Britta Swayn |      8 |            4 |
|  United States |        An Heidelber |     12 |            5 |
|  United States |        Maris Padill |     21 |            6 |
|  United States |     Rachele Italian |     21 |            7 |
|  United States |   Jacquiline Speake |     28 |            8 |
|  United States |      Hipolito Elami |     45 |            9 |
|  United States |          Earl Sayle |     65 |           10 |
|  United States |       Georgeann Ves |     66 |           11 |
|  United States |       Conchit Salli |     77 |           12 |

Схема выглядит следующим образом ( http://sqlfiddle.com/):

create table Country(
  id INTEGER NOT NULL GENERATED ALWAYS AS IDENTITY,
  name varchar(255),
  PRIMARY KEY (id)
);

create table Team(
  id INTEGER NOT NULL GENERATED ALWAYS AS IDENTITY,
  country_id int not null,
  PRIMARY KEY (id),
  FOREIGN KEY (country_id) REFERENCES Country(id)
);

create table Participant(
  id INTEGER NOT NULL GENERATED ALWAYS AS IDENTITY,
  team_id int not null,
  name varchar(100),
  points int,
  PRIMARY KEY (id),
  FOREIGN KEY (team_id) REFERENCES Team(id)
);

Вот что я попробовал:

select
    Country.name,
    Participant.name,
    Participant.points,
    ROW_NUMBER() OVER(order by Country.name, Participant.points) as country_rank
from Country
join Team
    on Country.id = Team.country_id
join Participant
    on Team.id = Participant.team_id;

Но согласно Apache Derby Doco, оператор OVER() не принимает никаких аргументов.

У кого-нибудь есть способ добиться звания страны?

3 ответа

Решение

запрос

SELECT c.name AS Country,
       p.name AS Participant,
       p.points AS Points,
       (SELECT COUNT(*)
        FROM Participant p2
        JOIN Team t2 ON p2.team_id = t2.id
        WHERE t2.country_id = t.country_id
          AND (p2.points < p.points
               OR p2.points = p.points AND p2.name <= p.name)) AS country_rank
FROM Country c
JOIN Team t ON c.id = t.country_id
JOIN Participant p ON t.id = p.team_id
ORDER BY c.name, p.points, p.name;

объяснение

Простой подвыбор ANSI-SQL может использоваться для выполнения той же работы, подсчитывая количество записей для участников в той же стране с более низким баллом или с таким же баллом и именем, которое в алфавитном порядке не выше.

демонстрация

Демонстрация SQL скрипки

Рассмотрим SQL-запрос не-оконной функции, в котором используется коррелированный подзапрос совокупного количества. Поскольку столбец группы (Country.name) не находится в той же таблице, что и критерии ранга (Participant.points), нам нужно выполнить те же соединения в подзапросе, но переименовать псевдонимы таблиц, чтобы правильно сравнить внутренние и внешние запросы.

Сейчас, конечно, в идеальном мире, который был бы этим, но теперь мы должны учитывать связанные моменты. Следовательно, другой очень похожий подзапрос (для прерывателя связей) используется для добавления в первый подзапрос. Этот второй вложенный запрос совпадает с внутренним и внешним запросом Country.name а также Participant.points но занимает в алфавитном порядке Participant.name,

SELECT
    Country.name AS Country,
    Participant.name AS Participant,
    Participant.points,
    (SELECT Count(*) + 1
       FROM Country subC
      INNER JOIN Team subT
              ON subC.id = subT.country_id
      INNER JOIN Participant subP
              ON subT.id = subP.team_id
      WHERE subC.name = Country.name
        AND subP.points < Participant.points) 

     +

    (SELECT Count(*)
       FROM Country subC
      INNER JOIN Team subT
              ON subC.id = subT.country_id
      INNER JOIN Participant subP
              ON subT.id = subP.team_id
      WHERE subC.name = Country.name
        AND subP.points = Participant.points
        AND subP.name < Participant.name)  As country_rank

FROM Country
INNER JOIN Team
        ON Country.id = Team.country_id
INNER JOIN Participant
        ON Team.id = Participant.team_id
ORDER BY Country.name, Participant.points;

Все, что вам нужно добавить, это разделить по странам, и это должно дать вам то, что вам нужно.

SELECT
Country.name,
Participant.name,
Participant.points,
ROW_NUMBER() OVER(PARTITION BY country order by Country.name, Participant.points) as country_rank
from Country
join Team
  on Country.id = Team.country_id
join Participant
  on Team.id = Participant.team_id;
Другие вопросы по тегам