How to get a single player's rank based on the score
Recently, a friend of mine approached me asking an advise for the task of getting the rank based on the score. When we have a list of, say, players, each having some score points, naturally these points are used to range the players on the scoreboard. Although to create a scoreboard is a trivial task - just order the table by the score,
SELECT * FROM players ORDER BY score DESC
to get a single player's rank based on their score is not that simple.
Surprisingly, there are no good solutions on Stack Overflow (the top result from Google for me for example) . Instead, some SQL monsters are offered, using variables, DBMS-specific functions like RANK(), derived tables and such. But I felt that the solution could be much simpler and could be done using just basic SQL operators.
The basic principle
The basic solution, given the score is known, is ridiculously simple: we just have to count all players placed on the scoreboard before. Naturally, it means that we need to count all players with a score higher than the given value:
SELECT count(*) FROM players WHERE score > ?
Bingo?..
Not yet. This query is giving us just the number of players before us, whereas our rank is one point bigger. So the proper query would be
SELECT count(*)+1 FROM players WHERE score > ?
Bingo?..
Not yet.
Adding another field
Well, there is a pitfall: it may happen that some users will share the same score. On the one hand, we can give such users the same rank. That's possible, but it will make the query too complex. Besides, such an approach is rarely used because people prefer to be certain about their rank. To make things straight, an additional field is often used in the ORDER BY clause when displaying a scoreboard, to make the resulting list consistent. It could be any field - the player's name or id for example. But it seems fair to use the time when the current score has been achieved - the first to come is higher on the scoreboard:
SELECT * FROM players ORDER BY score DESC, score_updated ASC
Therefore, to display the rank, we need to take into consideration this another field as well
The first thing that comes to mind (well, at least came to my mind, I have to confess) is simply to add the new field to the condition:
SELECT count(*)+1 FROM players WHERE score > ? AND score_updated < ?
Well, obviously, it doesn't work. It will cut off all the players who have a higher score but also a lesser update time. We need to separate these conditions, so they won't interfere. It is usually done with OR
operator. But yeah, simply changing AND to OR will just wreck havoc in the results. By this time I started visualizing the players table as a sort of line, where we have a part with all the players with higher score and a part with mates sharing the same score, like this
me
===================|===*====
superiors peers
and the solution instantly sprang to my mind: we just need to count these two parts separately:
SELECT count(*)+1 FROM players WHERE score > ? OR (score = ? AND score_updated < ?)
the leftmost condition is to select all the superiors and the second condition to count all the peers who have lesser update time. Bingo!
Getting the rank along with other data
The last stroke.
Imagine we are displaying the player's account page and among other stats have to provide their rank from the scoreboard. All we know is the player's id. Of course we could get the desired info in two queries - first getting the score and the update time, and then getting the rank. But it smells. That's a user's data all the same, why not to get it in a single query? So it makes the question, how to get the score based on id. The most obvious solution is a sub-query, like
SELECT count(*)+1 FROM players WHERE score > (SELECT score FROM players WHERE id = ?)
but that's just for one condition and we have two! It will make the SQL monster we are trying to avoid. Luckily, there is another way to get the data from the same row, by joining the table to itself:
SELECT p1.*, count(p2.id)+1 as rank FROM players p1
JOIN players p2 on p2.score > p1.score or (p2.score = p1.score AND p2.score_updated < p1.score_updated)
WHERE p1.id = 7393
this latter query is just a familiar LEFT JOIN with a count() - what could be simpler?
Note: an index for the score
is not to be forgotten! This field is heavily used in this query and without an index it will take for the database to rake through all the table. Whereas using an index it will only take to calculate the entries in the index, which is incomparable faster.
Related articles:
- Relative and absolute paths, in the file system and on the web server.
- PHP error reporting
- Do you really need to check for both isset() and empty() at the same time?
- What's wrong with popular articles telling you that foo is faster than bar?
- MVC in simpler terms or the structure of a modern web-application
- Articles
- Do you abuse the null coalescing operator (and isset/empty as well)?
- Operator precedence or how does 'or die()' work.
- Why should I use prepared statements if escaping is safe?
- Numerical strings comparison
- Do generators really reduce the memory usage?
- How to make Stack Overflow a nice place
- Iterating over array getting look-ahead items along
- How to debug small PHP programs
Add a comment
Please refrain from sending spam or advertising of any sort.
Messages with hyperlinks will be pending for moderator's review.
Markdown is now supported:
>
before and an empty line after for a quote