如何使用WHERE子句和UNION ALL在MySQL中排名

时间:2010-08-03 16:01:57

标签: mysql select ranking

我需要一些MySQL查询的帮助。我正在尝试使用WHERE子句对参与者进行排名。所以,我想分别对新手,中间人和经验进行排名。例如:

Rank Name    Grade  Type
----------------------------------
1    Bob     98     Novice 
2    Jill    88     Novice 
3    Jimmy   42     Novice 
1    Mark    87     Intermediate 
2    Scott   85     Intermediate
3    Jim     77     Intermediate 
1    Jane    90     Advanced 
2    John    89     Advanced 
3    Josh    87     Advanced

我试过了:

SET @rank=0;

(SELECT @rank:=@rank+1 AS rank, name, grade, type FROM myTable WHERE type='novice' ORDER BY grade DESC)
UNION ALL
(SELECT @rank:=@rank+1 AS rank, name, grade, type FROM myTable WHERE type='intermediate' ORDER BY grade DESC)
UNION ALL
(SELECT @rank:=@rank+1 AS rank, name, grade, type FROM myTable WHERE type='experienced' ORDER BY grade DESC)

我想我需要以某种方式重新设定等级。也许我有另一个问题?

2 个答案:

答案 0 :(得分:1)

使用:

SELECT t.name,
       t.grade,
       t.type,
       (SELECT COUNT(*)
          FROM YOUR_TABLE x
         WHERE x.type = t.type
           AND x.grade >= t.grade) AS rank
 FROM YOUR_TABLE t

要处理两个等级列,请使用:

SELECT t.name,
       t.grade1,
       t.grade2,
       t.type,
       (SELECT COUNT(*)
          FROM YOUR_TABLE x
         WHERE x.type = t.type
           AND (x.grade1 + x.grade2) >= (t.grade1 + t.grade2)) AS rank
 FROM YOUR_TABLE t

我如何将新手和中间人分别排在一起和高级?

使用:

SELECT y.*,
       (SELECT COUNT(*)
          FROM (SELECT *, 
                       CASE t.type
                         WHEN 'Advanced' THEN t.type
                         ELSE 'Non-Advanced'
                       END AS group_type
                  FROM YOUR_TABLE) x
         WHERE x.group_type = y.group_type
           AND (x.grade1 + x.grade2) >= (y.grade1 + y.grade2)) AS rank
  FROM (SELECT t.name,
               t.grade1,
               t.grade2,
               t.type,
               CASE t.type
                 WHEN 'Advanced' THEN t.type
                 ELSE 'Non-Advanced'
               END AS group_type
          FROM YOUR_TABLE t) y

答案 1 :(得分:0)

您可以使用更复杂的ORDER BY子句一次性完成。

SELECT @rank:=@rank+1 AS rank, name, grade, type 
FROM myTable 
ORDER BY 
    case type when 'novice'       then 1
              when 'intermediate' then 2 
              when 'experienced'  then 3 end,
    grade DESC
;

为了使它更好,您可以使用Type表来存储排序顺序,然后添加连接,然后只需按type.sort_order排序,然后按等级排序,例如

SELECT @rank:=@rank+1 AS rank, name, grade, type 
FROM myTable 
JOIN Types ON myTable.type = Types.type 
ORDER BY Types.sort_order, grade DESC;