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

 更新时间:2025年09月02日 10:21:47   作者:舒一笑不秃头  
本文主要介绍了利用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();
    }
}

到此这篇关于利用Mybatis自定义排序规则实现复杂排序的文章就介绍到这了,更多相关Mybatis 复杂排序内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 详解Java数据结构之平衡二叉树

    详解Java数据结构之平衡二叉树

    平衡二叉树(Balanced Binary Tree)又被称为AVL树(有别于AVL算法),且具有以下性质:它是一 棵空树或它的左右两个子树的高度差的绝对值不超过1,并且左右两个子树都是一棵平衡二叉树。本文将详解介绍一下平衡二叉树的原理与实现,需要的可以参考一下
    2022-02-02
  • Springboot的Mapper中添加新的SQL语句方法详解

    Springboot的Mapper中添加新的SQL语句方法详解

    在如今的软件开发界,Spring Boot可是非常受欢迎的框架哦,尤其是在微服务和RESTful API的构建上,下面给大家介绍我们如何为Spring Boot项目中的Mapper添加新的SQL语句吧,感兴趣的朋友一起看看吧
    2025-04-04
  • 一文详解SpringMVC中的@RequestMapping注解

    一文详解SpringMVC中的@RequestMapping注解

    @RequestMapping是一个用于映射HTTP请求到处理方法的注解,在Spring框架中使用,它可以用于控制器类和处理方法上,用来指定处理不同URL路径的请求,并定义请求的方法等,本文小编将给大家详细的介绍一下SpringMVC中的@RequestMapping注解,需要的朋友可以参考下
    2023-08-08
  • SpringBoot下载文件的正确解法方式

    SpringBoot下载文件的正确解法方式

    这篇文章主要给大家介绍了关于SpringBoot下载文件的正确解法方式,SpringBoot是一款流行的框架,用于开发Web应用程序,在使用SpringBoot构建Web应用程序时,可能需要实现文件下载的功能,需要的朋友可以参考下
    2023-08-08
  • SpringBoot中实现多数据源连接和切换的方案

    SpringBoot中实现多数据源连接和切换的方案

    在Spring Boot中,通过AbstractRoutingDataSource实现多数据源连接是一种常见的做法,这种技术允许你在运行时动态地切换数据源,从而支持对多个数据库的操作,本文给大家介绍了SpringBoot中实现多数据源连接和切换的方案,需要的朋友可以参考下
    2024-11-11
  • SpringCloud手写Ribbon实现负载均衡

    SpringCloud手写Ribbon实现负载均衡

    这篇文章主要介绍了SpringCloud手写Ribbon实现负载均衡的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-01-01
  • Spring Data JPA 建立表的联合主键

    Spring Data JPA 建立表的联合主键

    这篇文章主要介绍了Spring Data JPA 建立表的联合主键。本文详细的介绍了2种方式,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2019-04-04
  • Java读取文件方法汇总

    Java读取文件方法汇总

    这篇文章主要为大家详细介绍了Java读取文件方法,按字节读取文件内容、按字符读取文件内容、随机读取文件内容等,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-12-12
  • MyBatis3.X复杂Sql查询的语句

    MyBatis3.X复杂Sql查询的语句

    这篇文章主要介绍了MyBatis3.X复杂Sql查询的相关资料,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-04-04
  • Java计算数学表达式代码详解

    Java计算数学表达式代码详解

    这篇文章主要介绍了Java计算数学表达式代码详解,具有一定借鉴价值,需要的朋友可以了解下。
    2017-12-12

最新评论