勉強したことのメモ

Webエンジニア / プログラマが勉強したことのメモ。

MySQLのRANK及びDENSE_RANK関数を用いてランキングを実装する方法

  MySQL データベース

ランキング機能を実装する際、上位〇位までのデータをMySQLから持ってきて順位付けはPHP側で行っていた。ただ、MySQL8からはRANK及びDENSE_RANKという関数が実装され、こちらを用いることでMySQLのみで対応できるみたい。以下に対応方法をメモ。

 

リファレンス

RANK

https://dev.mysql.com/doc/refman/8.0/ja/window-function-descriptions.html#function_rank

DENSE_RANK

https://dev.mysql.com/doc/refman/8.0/ja/window-function-descriptions.html#function_dense-rank

 

テーブル情報

テーブル構造

mysql> SHOW COLUMNS FROM `test_table`;
+-------+--------------+------+-----+---------+----------------+
| Field | Type         | Null | Key | Default | Extra          |
+-------+--------------+------+-----+---------+----------------+
| id    | int          | NO   | PRI | NULL    | auto_increment |
| name  | varchar(255) | NO   |     | NULL    |                |
| score | int          | NO   |     | NULL    |                |
+-------+--------------+------+-----+---------+----------------+

データ内容

mysql> SELECT * FROM `test_table`;
+----+-----------+-------+
| id | name      | score |
+----+-----------+-------+
|  1 | 鈴木      |    50 |
|  2 | 山田      |    60 |
|  3 | 佐々木    |    70 |
|  4 | 佐藤      |    40 |
|  5 | 渡辺      |    60 |
|  6 | 高橋      |    80 |
+----+-----------+-------+

 

実装方法

同率順位があった場合、次の順位は+2とする場合(RANK)

SELECT *, RANK() OVER (ORDER BY `score`) AS `rank` 
FROM `test_table`;

+----+-----------+-------+------+
| id | name      | score | rank |
+----+-----------+-------+------+
|  4 | 佐藤      |    40 |    1 |
|  1 | 鈴木      |    50 |    2 |
|  2 | 山田      |    60 |    3 |
|  5 | 渡辺      |    60 |    3 |
|  3 | 佐々木    |    70 |    5 |
|  6 | 高橋      |    80 |    6 |
+----+-----------+-------+------+

同率順位があっても次の順位は+1とする場合(DENSE_RANK)

SELECT *, DENSE_RANK() OVER (ORDER BY `score`) AS `rank` 
FROM `test_table`;

+----+-----------+-------+------+
| id | name      | score | rank |
+----+-----------+-------+------+
|  4 | 佐藤      |    40 |    1 |
|  1 | 鈴木      |    50 |    2 |
|  2 | 山田      |    60 |    3 |
|  5 | 渡辺      |    60 |    3 |
|  3 | 佐々木    |    70 |    4 |
|  6 | 高橋      |    80 |    5 |
+----+-----------+-------+------+

 - MySQL データベース

  関連記事

MySQLでSELECT時に数値を3桁ずつのカンマ区切りに変換する方法

MySQLで商品価格のような数値の値を3桁ずつのカンマ区切りで取り出したいという ...

MySQLでJOINの高速化

JOINで無茶苦茶悩んだのでメモ。やりたかった事は、MySQLで テーブルAには ...

MySQLでカラム内容によって条件分岐させたい場合の対応方法

MySQLでカラムの内容によって条件を分岐させたかった。またPHPで処理するので ...

MySQLで直近に挿入したオートインクリメントの値と次回挿入する値を取得する方法

phpとmysqliを使っている中で次回挿入するオートインクリメントの値と、前回 ...

MySQLのREGEXPで半角記号を検索したい場合の対応方法

MySQLのREGEXPで半角記号を検索したい場合\\を付けてエスケープしないと ...