For your current table of 1,700
rows your solution is OK.
If you will have like 100,000
of rows, the DISTINCT
may become inefficient.
Here's the article in my blog that shows how to do it efficiently:
This solution employs an index on name
. It will jump over the index keys, selecting each first letter at most once.
First, you'll need to create a function:
CREATE FUNCTION fn_get_next_code(initial INT) RETURNS INT
NOT DETERMINISTIC
READS SQL DATA
BEGIN
DECLARE _next VARCHAR(200);
DECLARE EXIT HANDLER FOR NOT FOUND RETURN NULL;
SELECT ORD(SUBSTRING(name, 1, 1))
INTO _next
FROM t_names
WHERE name >= CHAR(initial + 1)
ORDER BY
name
LIMIT 1;
RETURN _next;
END
This function, given a code of a starting letter, returns the first starting letter next to the given from your table.
Second, use this function in a query:
SELECT CHAR(@r) AS starting,
@r := fn_get_next_letter(@r) AS next
FROM (
SELECT @r := ORD(LEFT(MIN(name), 1))
) vars, mytable
WHERE @r IS NOT NULL
On each iteration, session variable @r
will skip to the next starting letter using an index.
This will be very fast, but it pays for itself only if you have hundreds of thousands of rows.
Otherwise just use DISTINCT
.