当前位置: 首页 > news >正文

利用Mybatis自定义排序规则实现复杂排序

场景分析

本次需要实现规则是根据用户点击目录左侧——弹出排序选项,文件名A-Z:英文A-Z → 中文A-Z → 数字0-9 ,文件名Z-A:与A-Z相反,最近更新时间:文件夹按文件夹修改时间,文档按文档修改时间。图片直达链接

如何实现?

1.改造接口增加参数

    /*** 排序类型: NAME_ASC(文件名A-Z), NAME_DESC(文件名Z-A), TIME_DESC(最近更新时间)*/@Schema(description = "排序类型: NAME_ASC(文件名A-Z), NAME_DESC(文件名Z-A), TIME_DESC(最近更新时间)")private String sortType;

2.修改默认查询子集排序规则

// 排序 - 默认按sort字段排序,如果需要其他排序规则会在数据库层处理
this.children.sort(Comparator.comparing(KbPageTreeResp::getSort, Comparator.nullsLast(Comparator.naturalOrder())));

这个表达式是一个复合比较器,用于处理包含 null 值的排序场景。

1. Comparator.naturalOrder()
  • 这是一个基础比较器,用于对实现了 Comparable 接口的对象进行自然排序
  • 对于 Integer 类型,自然排序就是数值升序(1, 2, 3, 4…)
  • 相当于调用对象的 compareTo() 方法
2. Comparator.nullsLast(...)
  • 这是一个装饰器比较器,用于处理 null
  • nullsLast 表示:null 值排在最后
  • 它接受一个内部比较器作为参数,用于比较非 null
3. 组合效果

Comparator.nullsLast(Comparator.naturalOrder()) 的排序规则是:

  1. null:按照自然排序(升序)排列
  2. null:统一排在最后面
  3. 混合情况:非 null 值在前面按升序排列,null 值在最后
4. 实际例子

假设 sort 字段的值有:[3, null, 1, null, 2]

排序后的结果将是:[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. 在您代码中的应用

KbPageTreeResp 类中,这个比较器用于:

this.children.sort(Comparator.comparing(KbPageTreeResp::getSort, Comparator.nullsLast(Comparator.naturalOrder())));
  • Comparator.comparing(KbPageTreeResp::getSort, ...) 提取每个对象的 sort 字段进行比较
  • 如果某些页面的 sort 字段为 null,这些页面会被排在最后
  • 其他有 sort 值的页面按照数值升序排列
6. 为什么需要处理 null

在实际业务中,页面的 sort 字段可能:

  • 新创建的页面还没有设置排序值(null
  • 某些页面被明确设置为不参与排序(null
  • 数据迁移或其他原因导致的空值

使用 nullsLast 可以确保程序不会因为 null 值而抛出 NullPointerException,同时提供合理的排序行为。

3.自定义排序规则

动态构建 SQL 排序语句(ORDER BY)**的 Java 方法。它的核心作用是根据**文件类型、标题、时间等多个维度,对一组文件/文件夹进行复合排序,并且支持多种排序策略(升序、降序、按时间、按名称等)。

  /*** 构建ORDER BY语句* 文件夹和文件分别排序,文件夹在前,文件在后* xxxxx节点始终置顶* * @return ORDER BY语句*/private String buildOrderByClause() {StringBuilder orderBy = new StringBuilder();// 1. xxxxxx节点置顶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. 根据排序类型进行排序if (StrUtil.isNotBlank(this.sortType)) {switch (this.sortType.toUpperCase()) {case "NAME_ASC":// 文件名A-Z:英文A-Z → 中文A-Z → 数字0-9// 使用ASCII码和字符判断来实现优先级排序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) ") // 数字.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) ") // 数字.append("ELSE CONCAT('4', title) ") // 其他字符.append("END DESC");break;case "TIME_DESC":// 最近更新时间:文件夹按文件夹修改时间,文档按文档修改时间orderBy.append("modifier_time DESC");break;default:// 默认排序:修改时间降序 + sort升序 + 标题升序orderBy.append("modifier_time DESC, sort ASC, title ASC");break;}} else {// 默认排序:修改时间降序 + sort升序 + 标题升序orderBy.append("modifier_time DESC, sort ASC, title ASC");}return orderBy.toString();}
}

http://www.dtcms.com/a/362055.html

相关文章:

  • UBUNTU之Onvif开源服务器onvif_srvd:1、编译
  • Java synchronized 关键字详解
  • JAVA后端开发——forEach 与方法引用(::)详解
  • 剧想智读项目展示
  • 针对redis中的热数据该怎么处理
  • Qt 项目文件(.pro)中添加 UI 文件相关命令
  • 【系规伴学】云资源规划核心知识点解析
  • vcruntime140.dll丢失解决办法
  • pandas自学笔记16 pandas可视化
  • 继承体系中的隐藏机制解析(继承中的作用域)
  • Unity图集 SpriteAltas 打包探究
  • [界面通过zmq请求调用指定动态库函数(三)]将zmq请求集成二次开发接口
  • Android 文件下载 FileDownloader
  • TypeScript交叉类型、重复属性、类型断言和as const详解
  • 光颉科技)Viking)的CS25FTFR009 1225 0.009R/9mR 3W电阻介绍-华年商城
  • 【笔记】float类型的精度有限,无法精确表示123456.6789
  • 【Audio】静音或振动模式下重复来电响铃
  • Java Web项目后台管理系统之内容管理仿写(三):图片上传
  • 使用 BayesFlow 通过神经网络简化贝叶斯推断(一)
  • C扩展4:X宏(X-MACRO)
  • JS循环机制
  • IS-IS的原理
  • Java超卖问题
  • MySQL安装与使用指南
  • 【读论文】量子关联增强双梳光谱技术
  • 力扣404 代码随想录Day15 第三题
  • 故障排查指南:理解与解决 “No route to host“ 错误
  • NOSQL——Redis
  • MySQL基础知识保姆级教程(四)视图与约束
  • 浅谈中断控制器:从 IRQ 到 IRR、IMR、In-Service Register