Я использую MYSQL для расчета TOP-N по каждому предмету в таблице ученика. - PullRequest
0 голосов
/ 07 июня 2018

У меня есть таблица ученика, в которой указаны student_id, балл и предмет

CREATE TABLE IF NOT EXISTS students 
(student_id INT(3),   subject ,VARCHAR(45),   score INT(3) );

Введены следующие данные:

insert into students values(1,'math',70);
insert into students values(1,'science',71);
insert into students values(1,'history',72);
insert into students values(1,'english',73);
insert into students values(1,'kannada',74);
insert into students values(3,'math',50);
insert into students values(3,'science',51);
insert into students values(3,'history',52);
insert into students values(3,'english',53);
insert into students values(3,'kannada',54);
insert into students values(2,'math',60);
insert into students values(2,'science',61);
insert into students values(2,'history',62);
insert into students values(2,'english',63);
insert into students values(2,'kannada',64);

После использования запроса я получаю требуемый результат,

select student_id,score,subject
    from
        (select @prev := '', @n:=0) init
    join
        (select @n := if(subject != @prev , 1, @n+1) as n,
                 @prev := subject,
                 student_id,score,subject from students
                 order by
                 subject asc, 
                 score desc
        ) x
        where n<=2
        order by subject, score desc;

Я просто не понимаю, как это работает, зачем нужно объединение?Это подзапрос?Будут ли операторы в предложении from выполняться для каждой строки данных?Кто-нибудь, пожалуйста, объясните мне это.Я учу SQL.

Примечание: я нашел этот запрос похожим на этот онлайн, и я просто адаптировал его под свои требования, это не моя работа.

1 Ответ

0 голосов
/ 07 июня 2018

Соединение просто необходимо, чтобы вы могли инициализировать переменные @prev и @n в запросе.Это должно быть сделано отдельно от запроса, который вы пытаетесь отфильтровать.Вместо этого вы могли бы сделать это до запроса, но это объединит все в один автономный запрос.

SET @prev = '';
SET @n = 0;
SELECT student_id, score, subject
FROM 
    (select @n := if(subject != @prev , 1, @n+1) as n,
             @prev := subject,
             student_id,score,subject from students
             order by
             subject asc, 
             score desc
    ) x
where n<=2
order by subject, score desc;

В этом случае используется подзапрос, так что вы можете выбрать нужные строки n <= 2.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...