Mysql 高阶自定义排序
(一)FIELD函数实现
FIELD()函数返回值列表中值的索引位置,语法:FIELD(value, val1, val2, val3, ...),简单的说:改函数就是返回value在集合(val1,val2,val3,...)中对应的索引位置,结合上面的需求,是不是可以这样来使用,学生表stage值就映射函数的value,(初中、高总、大学)就映射函数的值列表集合,如果学生.stage=初中,返回值不是就是1;学生.stage=高中,返回值不是就是2;学生.stage=大学,返回值不是就是3,在对结果值排一个序,不就满足要求了嘛,话不多说上代码:
SELECT id,name,stage FROM `user` ORDER BY FIELD(stage,"初中","高中","大学") asc;

(二)LOCATE函数实现
FIELD(substr,str,pos)函数返回字符串substr在str中第一次出现的位置,pos可空,表示开始坐标,例如:
SELECT LOCATE("初中","初中,高中,大学");--返回1
SELECT LOCATE("高中","初中,高中,大学");--返回2
SELECT LOCATE("大学","初中,高中,大学");--返回3
这不就正好满足上述需求嘛,话不多说上代码:
SELECT id,name,stage FROM `user` ORDER BY LOCATE(stage,"初中,高中,大学") asc;

(三)INSTR函数实现
INSTR(str,substr)函数返回字符串str中substr第一次出现的位置,例如:
SELECT INSTR("初中,高中,大学","初中");-- 返回1
SELECT INSTR("初中,高中,大学","高中");-- 返回2
SELECT INSTR("初中,高中,大学","大学");-- 返回3
这不就正好满足上述需求嘛,话不多说上代码:
SELECT id,name,stage FROM `user` ORDER BY INSTR("初中,高中,大学",stage) asc;

(四)CASE WHEN语句实现
CASE WHEN简单的说就是根据不同的条件将其转换为指定的值,比如:根据stage 的不同值转换为对应的数字,在对结果排序满足要求了,话不多说上代码:
SELECT id,name,stage FROM `user` ORDER BY
CASE stage
WHEN "初中" THEN 1
WHEN "高中" THEN 2
WHEN "大学" THEN 3
ELSE 4 END asc;

(五)数据表字典实现
数据表字典思路就是需要单独建立一个stage 的值映射对应关系表,最后通过关联查询,并对根据映射值排序,正好能够满足上述要求,话不多说上代码:
SELECT u.id,u.name,u.stage FROM `user` as u
JOIN stage_dic as sd on u.stage = sd.stage
ORDER BY sd.code asc;

通过上面的5种排序,都能够解决问题引入提到的自定义排序需求。这5种排序,既有相同的应用场景,也有各自不同的应用场景,在实际使用中还需要根据实际情况选择不同的排序方式,下面对其做一个简单的总结:
- 针对简单枚举值皆可实现,数字表字典更灵活
- LOCATE与INSTR最终效果一致,参数颠倒,INSTR区分大小写,LOCATE不区分大小写
- 函数虽好,使用需要谨慎,能够在存储时就处理好数据,就不要在查询时通过函数计算来解