利用Mybatis自定義排序規(guī)則實(shí)現(xiàn)復(fù)雜排序
場景分析
本次需要實(shí)現(xiàn)規(guī)則是根據(jù)用戶點(diǎn)擊目錄左側(cè)——彈出排序選項(xiàng),文件名A-Z:英文A-Z → 中文A-Z → 數(shù)字0-9 ,文件名Z-A:與A-Z相反,最近更新時(shí)間:文件夾按文件夾修改時(shí)間,文檔按文檔修改時(shí)間。

如何實(shí)現(xiàn)?
1.改造接口增加參數(shù)
/**
* 排序類型: NAME_ASC(文件名A-Z), NAME_DESC(文件名Z-A), TIME_DESC(最近更新時(shí)間)
*/
@Schema(description = "排序類型: NAME_ASC(文件名A-Z), NAME_DESC(文件名Z-A), TIME_DESC(最近更新時(shí)間)")
private String sortType;
2.修改默認(rèn)查詢子集排序規(guī)則
// 排序 - 默認(rèn)按sort字段排序,如果需要其他排序規(guī)則會(huì)在數(shù)據(jù)庫層處理 this.children.sort(Comparator.comparing(KbPageTreeResp::getSort, Comparator.nullsLast(Comparator.naturalOrder())));
這個(gè)表達(dá)式是一個(gè)復(fù)合比較器,用于處理包含 null 值的排序場景。
1.Comparator.naturalOrder()
- 這是一個(gè)基礎(chǔ)比較器,用于對(duì)實(shí)現(xiàn)了
Comparable接口的對(duì)象進(jìn)行自然排序 - 對(duì)于
Integer類型,自然排序就是數(shù)值升序(1, 2, 3, 4...) - 相當(dāng)于調(diào)用對(duì)象的
compareTo()方法
2.Comparator.nullsLast(...)
- 這是一個(gè)裝飾器比較器,用于處理
null值 nullsLast表示:將null值排在最后- 它接受一個(gè)內(nèi)部比較器作為參數(shù),用于比較非
null值
3. 組合效果
Comparator.nullsLast(Comparator.naturalOrder()) 的排序規(guī)則是:
- 非
null值:按照自然排序(升序)排列 null值:統(tǒng)一排在最后面- 混合情況:非
null值在前面按升序排列,null值在最后
4. 實(shí)際例子
假設(shè) sort 字段的值有:[3, null, 1, null, 2]
排序后的結(jié)果將是:[1, 2, 3, null, null]
// 示例代碼 List<Integer> sorts = Arrays.asList(3, null, 1, null, 2); sorts.sort(Comparator.nullsLast(Comparator.naturalOrder())); System.out.println(sorts); // 輸出: [1, 2, 3, null, null]
5. 在您代碼中的應(yīng)用
在 KbPageTreeResp 類中,這個(gè)比較器用于:
this.children.sort(Comparator.comparing(KbPageTreeResp::getSort,
Comparator.nullsLast(Comparator.naturalOrder())));
Comparator.comparing(KbPageTreeResp::getSort, ...)提取每個(gè)對(duì)象的sort字段進(jìn)行比較- 如果某些頁面的
sort字段為null,這些頁面會(huì)被排在最后 - 其他有
sort值的頁面按照數(shù)值升序排列
6. 為什么需要處理null?
在實(shí)際業(yè)務(wù)中,頁面的 sort 字段可能:
- 新創(chuàng)建的頁面還沒有設(shè)置排序值(
null) - 某些頁面被明確設(shè)置為不參與排序(
null) - 數(shù)據(jù)遷移或其他原因?qū)е碌目罩?/li>
使用 nullsLast 可以確保程序不會(huì)因?yàn)?null 值而拋出 NullPointerException,同時(shí)提供合理的排序行為。
3.自定義排序規(guī)則
動(dòng)態(tài)構(gòu)建 SQL 排序語句(ORDER BY)*的 Java 方法。它的核心作用是根據(jù)*文件類型、標(biāo)題、時(shí)間等多個(gè)維度,對(duì)一組文件/文件夾進(jìn)行復(fù)合排序,并且支持多種排序策略(升序、降序、按時(shí)間、按名稱等)。
/**
* 構(gòu)建ORDER BY語句
* 文件夾和文件分別排序,文件夾在前,文件在后
* xxxxx節(jié)點(diǎn)始終置頂
*
* @return ORDER BY語句
*/
private String buildOrderByClause() {
StringBuilder orderBy = new StringBuilder();
// 1. xxxxxx節(jié)點(diǎn)置頂
orderBy.append("ORDER BY CASE WHEN title = 'xxxxxx' THEN 0 ELSE 1 END, ");
// 2. 文件夾在前,文件在后
orderBy.append("CASE WHEN type = 'FOLDER' THEN 0 ELSE 1 END, ");
// 3. 根據(jù)排序類型進(jìn)行排序
if (StrUtil.isNotBlank(this.sortType)) {
switch (this.sortType.toUpperCase()) {
case "NAME_ASC":
// 文件名A-Z:英文A-Z → 中文A-Z → 數(shù)字0-9
// 使用ASCII碼和字符判斷來實(shí)現(xiàn)優(yōu)先級(jí)排序
orderBy.append("CASE ")
.append("WHEN ASCII(SUBSTRING(title, 1, 1)) BETWEEN 65 AND 90 OR ASCII(SUBSTRING(title, 1, 1)) BETWEEN 97 AND 122 THEN CONCAT('1', title) ") // 英文字母
.append("WHEN ASCII(SUBSTRING(title, 1, 1)) > 127 THEN CONCAT('2', title) ") // 中文字符
.append("WHEN ASCII(SUBSTRING(title, 1, 1)) BETWEEN 48 AND 57 THEN CONCAT('3', title) ") // 數(shù)字
.append("ELSE CONCAT('4', title) ") // 其他字符
.append("END ASC");
break;
case "NAME_DESC":
// 文件名Z-A:與A-Z相反
orderBy.append("CASE ")
.append("WHEN ASCII(SUBSTRING(title, 1, 1)) BETWEEN 65 AND 90 OR ASCII(SUBSTRING(title, 1, 1)) BETWEEN 97 AND 122 THEN CONCAT('1', title) ") // 英文字母
.append("WHEN ASCII(SUBSTRING(title, 1, 1)) > 127 THEN CONCAT('2', title) ") // 中文字符
.append("WHEN ASCII(SUBSTRING(title, 1, 1)) BETWEEN 48 AND 57 THEN CONCAT('3', title) ") // 數(shù)字
.append("ELSE CONCAT('4', title) ") // 其他字符
.append("END DESC");
break;
case "TIME_DESC":
// 最近更新時(shí)間:文件夾按文件夾修改時(shí)間,文檔按文檔修改時(shí)間
orderBy.append("modifier_time DESC");
break;
default:
// 默認(rèn)排序:修改時(shí)間降序 + sort升序 + 標(biāo)題升序
orderBy.append("modifier_time DESC, sort ASC, title ASC");
break;
}
} else {
// 默認(rèn)排序:修改時(shí)間降序 + sort升序 + 標(biāo)題升序
orderBy.append("modifier_time DESC, sort ASC, title ASC");
}
return orderBy.toString();
}
}
到此這篇關(guān)于利用Mybatis自定義排序規(guī)則實(shí)現(xiàn)復(fù)雜排序的文章就介紹到這了,更多相關(guān)Mybatis 復(fù)雜排序內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
詳解Java數(shù)據(jù)結(jié)構(gòu)之平衡二叉樹
平衡二叉樹(Balanced?Binary?Tree)又被稱為AVL樹(有別于AVL算法),且具有以下性質(zhì):它是一?棵空樹或它的左右兩個(gè)子樹的高度差的絕對(duì)值不超過1,并且左右兩個(gè)子樹都是一棵平衡二叉樹。本文將詳解介紹一下平衡二叉樹的原理與實(shí)現(xiàn),需要的可以參考一下2022-02-02
Springboot的Mapper中添加新的SQL語句方法詳解
在如今的軟件開發(fā)界,Spring Boot可是非常受歡迎的框架哦,尤其是在微服務(wù)和RESTful API的構(gòu)建上,下面給大家介紹我們?nèi)绾螢镾pring Boot項(xiàng)目中的Mapper添加新的SQL語句吧,感興趣的朋友一起看看吧2025-04-04
一文詳解SpringMVC中的@RequestMapping注解
@RequestMapping是一個(gè)用于映射HTTP請(qǐng)求到處理方法的注解,在Spring框架中使用,它可以用于控制器類和處理方法上,用來指定處理不同URL路徑的請(qǐng)求,并定義請(qǐng)求的方法等,本文小編將給大家詳細(xì)的介紹一下SpringMVC中的@RequestMapping注解,需要的朋友可以參考下2023-08-08
SpringBoot中實(shí)現(xiàn)多數(shù)據(jù)源連接和切換的方案
在Spring Boot中,通過AbstractRoutingDataSource實(shí)現(xiàn)多數(shù)據(jù)源連接是一種常見的做法,這種技術(shù)允許你在運(yùn)行時(shí)動(dòng)態(tài)地切換數(shù)據(jù)源,從而支持對(duì)多個(gè)數(shù)據(jù)庫的操作,本文給大家介紹了SpringBoot中實(shí)現(xiàn)多數(shù)據(jù)源連接和切換的方案,需要的朋友可以參考下2024-11-11
SpringCloud手寫Ribbon實(shí)現(xiàn)負(fù)載均衡
這篇文章主要介紹了SpringCloud手寫Ribbon實(shí)現(xiàn)負(fù)載均衡的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01
Spring Data JPA 建立表的聯(lián)合主鍵
這篇文章主要介紹了Spring Data JPA 建立表的聯(lián)合主鍵。本文詳細(xì)的介紹了2種方式,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2019-04-04
Java計(jì)算數(shù)學(xué)表達(dá)式代碼詳解
這篇文章主要介紹了Java計(jì)算數(shù)學(xué)表達(dá)式代碼詳解,具有一定借鑒價(jià)值,需要的朋友可以了解下。2017-12-12

