java实现数据库排序功能|compare排序出现IllegalArgumentException: Comparison method violates its general contract

简介: java实现数据库排序功能|compare排序出现IllegalArgumentException: Comparison method violates its general contract

一、前言

java实现排序的时候,有时候会出现异常java.lang.IllegalArgumentException: Comparison method violates its general contract,


报这个异常的原因是代码里没有考虑对象o1和对象o2为Null的情况,


即当o1与o2都为null时两者大小如何判定呢;


当o1为null但o2不为null时两者大小又如何判定了呢,


同理当o2为null但o1不为null时两者大小又如何判定呢


所以代码里没有考虑上述情况的时候,就会出现Comparison method violates its general contract 异常了。


那怎么解决呢?


二、解决方法

第一种方法是在jvm启动参数中加上配置

-Djava.util.Arrays.useLegacyMergeSort=true

原因是因为JDK7中的Collections.Sort方法实现中,如果两个值是相等的,那么compare方法需要返回0,否则 可能 会在排序时抛错,而JDK6是没有这个限制的。

第二种方法是在代码中实现

下面分享一下我,java手把手实现数据库的排序规则


当有多个字段排序的时候,优先以第一个字段排序,如果第一个字段分出顺序就不考虑后面的字段;如果第一个字段相等,就再进行第二个字段的排序,以此类推


eg:a desc,b asc


数据库中表示先以a字段倒序排序,如果a字段拍出顺序,就不再进行排序,如果a字段值相等,就以 b字段升序排序。


java怎么实现上述功能呢?

java实现代码

我们先定义规则:


可以传入排序规则,多个规则用英文逗号分隔,冒号后面跟num还是text表示是用数字排序还是用字符串排序;如果冒号后面不跟num或者text,默认用字符串排序


eg:a.asc:num,b.desc,c.asc:text


表示a字段用数字类型升序排序,如果相等再用b字段字符串倒序排序,最后用c字段字符串升序排序

实现数据库排序的代码如下

import com.alibaba.fastjson.JSON;
import com.alibaba.fastjson.serializer.SerializerFeature;
import org.apache.commons.collections.CollectionUtils;
import org.apache.commons.lang3.StringUtils;
import java.util.*;
public class Testaaa {
    /**
     * 大于SORT_SIZE要排序
     */
    static Integer SORT_SIZE = 1;
    /**
     * 倒序排序
     */
    static String SORT_DESC = "desc";
    /**
     * 合并计算方式
     */
    static String COMBINECALC_FLAG = "1";
    /**
     * 该字段属于什么类型,数字还是字符串
     */
    static String TEXT_TYPE = "num";
    /**
     * 绝对值
     */
    static String TEXT_TYPE_ABS = "abs";
    /**
     * 冒号分隔符
     */
    static String COLON_SEPARATOR = ":";
    public static void main(String[] args) {
        List<Map<String, Object>> list = new ArrayList<>();
        Map<String, Object> map1 = new HashMap<>();
        map1.put("map","map1");
        map1.put("c","111");
        map1.put("d",null);
        list.add(map1);
        Map<String, Object> map2 = new HashMap<>();
        map2.put("map","map2");
        map2.put("c","333");
        map2.put("d","bbb");
        list.add(map2);
        Map<String, Object> map3 = new HashMap<>();
        map3.put("map","map3");
        map3.put("c","444");
        map3.put("d","aaa");
        list.add(map3);
        Map<String, Object> map4 = new HashMap<>();
        map4.put("map","map4");
        map4.put("c","222");
        map4.put("d",null);
        list.add(map4);
        Map<String, Object> map5 = new HashMap<>();
        map5.put("map","map5");
        map5.put("c",null);
        map5.put("d","ccc");
        list.add(map5);
        Map<String, Object> map6 = new HashMap<>();
        map6.put("map","map6");
        map6.put("c","444");
        map6.put("d",null);
        list.add(map6);
        Map<String, Object> map7 = new HashMap<>();
        map7.put("map","map7");
        map7.put("c",null);
        map7.put("d","cca");
        list.add(map7);
        Map<String, Object> map8 = new HashMap<>();
        map8.put("map","map8");
        map8.put("c",null);
        map8.put("d",null);
        list.add(map8);
        Map<String, Object> map9 = new HashMap<>();
        map9.put("map","map9");
        map9.put("c","555");
        map9.put("d",null);
        list.add(map9);
        Map<String, Object> map10 = new HashMap<>();
        map10.put("map","map10");
        map10.put("c","444");
        map10.put("d","eee");
        list.add(map10);
        String orderRule="c.desc:num,d.asc";
        sortResultList(orderRule, list);
        System.out.println(JSON.toJSONString(list, SerializerFeature.WriteMapNullValue));
    }
    private static void sortResultList(String orderRule, List<Map<String, Object>> resultList) {
        if (CollectionUtils.isNotEmpty(resultList) && StringUtils.isNotBlank(orderRule)) {
            List<String> ruleList = Arrays.asList(orderRule.split(","));
            // 对外部进行排序
            if (resultList.size() > SORT_SIZE) {
                sortList(ruleList, resultList);
            }
            // 对内部children进行排序
            for (Map<String, Object> stringObjectMap : resultList) {
                List<Map<String, Object>> children = (List<Map<String, Object>>) stringObjectMap.get("children");
                if (CollectionUtils.isNotEmpty(children) && children.size() > SORT_SIZE) {
                    sortList(ruleList, children);
                }
            }
        }
    }
    public static void sortList(List<String> ruleList, List<Map<String, Object>> result) {
        Collections.sort(result, new Comparator<Map<String, Object>>() {
            @Override
            public int compare(Map<String, Object> str1, Map<String, Object> str2) {
                return compareRule(ruleList, 0, str1, str2);
            }
        });
    }
    /**
     * 递归排序
     * @param ruleList
     * @param i
     * @param str1
     * @param str2
     * @return
     */
    private static Integer compareRule(List<String> ruleList, Integer i, Map<String, Object> str1, Map<String, Object> str2) {
        int sort = 0;
        int no = i;
        if (i < ruleList.size()) {
            // 排序规则:eg:date.asc:num,id.desc:text
            String rule = ruleList.get(i);
            // data asc:num 或者 id desc
            String[] typeArray = rule.split("[.]");
            if (str2.containsKey(typeArray[0]) && str2.get(typeArray[0]) != null
                    && StringUtils.isNotBlank(str2.get(typeArray[0]).toString())
                    && str1.containsKey(typeArray[0]) && str1.get(typeArray[0]) != null
                    && StringUtils.isNotBlank(str1.get(typeArray[0]).toString())) {
                // 该字段排序是用什么类型:数字还是字符串
                String[] textType = new String[2];
                if (typeArray[1].indexOf(COLON_SEPARATOR) != -1) {
                    textType = typeArray[1].split("[:]");
                } else {
                    textType[0] = typeArray[1];
                    textType[1] = "text";
                }
                // 按数字排序
                if (StringUtils.isNotBlank(textType[1]) && TEXT_TYPE.equalsIgnoreCase(textType[1])) {
                    if (SORT_DESC.equalsIgnoreCase(textType[0])) {
                        sort = Double.valueOf(str2.get(typeArray[0]).toString()).compareTo(Double.valueOf(str1.get(typeArray[0]).toString()));
                    } else {
                        sort = Double.valueOf(str1.get(typeArray[0]).toString()).compareTo(Double.valueOf(str2.get(typeArray[0]).toString()));
                    }
                } else if (StringUtils.isNotBlank(textType[1]) && TEXT_TYPE_ABS.equalsIgnoreCase(textType[1])) {
                    // 绝对值排序
                    double val1;
                    double val2;
                    if (SORT_DESC.equalsIgnoreCase(textType[0])) {
                        val1 = Double.parseDouble(str2.get(typeArray[0]).toString());
                        val2 = Double.parseDouble(str1.get(typeArray[0]).toString());
                    } else {
                        val1 = Double.parseDouble(str1.get(typeArray[0]).toString());
                        val2 = Double.parseDouble(str2.get(typeArray[0]).toString());
                    }
                    sort = Double.compare(Math.abs(val1), Math.abs(val2));
                } else {
                    // 字符串排序
                    // 倒序
                    if (SORT_DESC.equalsIgnoreCase(textType[0])) {
                        sort = str2.get(typeArray[0]).toString().compareTo(str1.get(typeArray[0]).toString());
                    } else {
                        // 正序
                        sort = str1.get(typeArray[0]).toString().compareTo(str2.get(typeArray[0]).toString());
                    }
                }
                // 如果第一个字段相等,按第二个字段排序
                if (sort == 0) {
                    no = no + 1;
                    sort = compareRule(ruleList, no, str1, str2);
                } else {
                    return sort;
                }
            } else {
                // 将null值往后面排
                if (
                    // 第一个值为null,第二个值不为null的情况
                        ((!str1.containsKey(typeArray[0])) ||
                                (str1.containsKey(typeArray[0]) && str1.get(typeArray[0]) == null) ||
                                (str1.containsKey(typeArray[0]) && str1.get(typeArray[0]) != null && StringUtils.isBlank(str1.get(typeArray[0]).toString()))) &&
                                (str2.containsKey(typeArray[0]) && str2.get(typeArray[0]) != null && StringUtils.isNotBlank(str2.get(typeArray[0]).toString()))) {
                    sort = 1;
                } else if (
                    // 第一个值不为null,第二个值为null的情况
                        ((!str2.containsKey(typeArray[0])) ||
                                (str2.containsKey(typeArray[0]) && str2.get(typeArray[0]) == null) ||
                                (str2.containsKey(typeArray[0]) && str2.get(typeArray[0]) != null && StringUtils.isBlank(str2.get(typeArray[0]).toString()))) &&
                                (str1.containsKey(typeArray[0]) && str1.get(typeArray[0]) != null && StringUtils.isNotBlank(str1.get(typeArray[0]).toString()))) {
                    sort = -1;
                } else {
                    // 第一个值和第二个值都为null情况
                    sort = 0;
                }
                // 如果第一个字段相等,按第二个字段排序
                if (sort == 0) {
                    no = no + 1;
                    sort = compareRule(ruleList, no, str1, str2);
                } else {
                    return sort;
                }
            }
        }
        return sort;
    }
}

运行结果:

相关文章
|
2天前
|
XML Java 数据库连接
性能提升秘籍:如何高效使用Java连接池管理数据库连接
在Java应用中,数据库连接管理至关重要。随着访问量增加,频繁创建和关闭连接会影响性能。为此,Java连接池技术应运而生,如HikariCP。本文通过代码示例介绍如何引入HikariCP依赖、配置连接池参数及使用连接池高效管理数据库连接,提升系统性能。
15 5
|
5天前
|
Java
Java 8 引入的 Streams 功能强大,提供了一种简洁高效的处理数据集合的方式
Java 8 引入的 Streams 功能强大,提供了一种简洁高效的处理数据集合的方式。本文介绍了 Streams 的基本概念和使用方法,包括创建 Streams、中间操作和终端操作,并通过多个案例详细解析了过滤、映射、归并、排序、分组和并行处理等操作,帮助读者更好地理解和掌握这一重要特性。
11 2
|
14天前
|
SQL Java 数据库连接
在Java应用中,数据库访问常成为性能瓶颈。连接池技术通过预建立并复用数据库连接,有效减少连接开销,提升访问效率
在Java应用中,数据库访问常成为性能瓶颈。连接池技术通过预建立并复用数据库连接,有效减少连接开销,提升访问效率。本文介绍了连接池的工作原理、优势及实现方法,并提供了HikariCP的示例代码。
30 3
|
14天前
|
存储 Java 关系型数据库
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践
在Java开发中,数据库连接是应用与数据交互的关键环节。本文通过案例分析,深入探讨Java连接池的原理与最佳实践,包括连接创建、分配、复用和释放等操作,并通过电商应用实例展示了如何选择合适的连接池库(如HikariCP)和配置参数,实现高效、稳定的数据库连接管理。
31 2
|
14天前
|
Java 数据库连接 数据库
如何构建高效稳定的Java数据库连接池,涵盖连接池配置、并发控制和异常处理等方面
本文介绍了如何构建高效稳定的Java数据库连接池,涵盖连接池配置、并发控制和异常处理等方面。通过合理配置初始连接数、最大连接数和空闲连接超时时间,确保系统性能和稳定性。文章还探讨了同步阻塞、异步回调和信号量等并发控制策略,并提供了异常处理的最佳实践。最后,给出了一个简单的连接池示例代码,并推荐使用成熟的连接池框架(如HikariCP、C3P0)以简化开发。
32 2
|
14天前
|
Java 数据库连接 数据库
深入探讨Java连接池技术如何通过复用数据库连接、减少连接建立和断开的开销,从而显著提升系统性能
在Java应用开发中,数据库操作常成为性能瓶颈。本文通过问题解答形式,深入探讨Java连接池技术如何通过复用数据库连接、减少连接建立和断开的开销,从而显著提升系统性能。文章介绍了连接池的优势、选择和使用方法,以及优化配置的技巧。
16 1
|
14天前
|
Java 数据库连接 数据库
Java连接池在数据库性能优化中的重要作用。连接池通过预先创建和管理数据库连接,避免了频繁创建和关闭连接的开销
本文深入探讨了Java连接池在数据库性能优化中的重要作用。连接池通过预先创建和管理数据库连接,避免了频繁创建和关闭连接的开销,显著提升了系统的响应速度和吞吐量。文章介绍了连接池的工作原理,并以HikariCP为例,展示了如何在Java应用中使用连接池。通过合理配置和优化,连接池技术能够有效提升应用性能。
30 1
|
14天前
|
监控 Java 数据库连接
在Java开发中,数据库连接管理是关键问题之一
在Java开发中,数据库连接管理是关键问题之一。本文介绍了连接池技术如何通过预创建和管理数据库连接,提高数据库操作的性能和稳定性,减少资源消耗,并简化连接管理。通过示例代码展示了HikariCP连接池的实际应用。
16 1
|
7天前
|
SQL 关系型数据库 MySQL
go语言数据库中mysql驱动安装
【11月更文挑战第2天】
20 4
|
4天前
|
SQL 关系型数据库 MySQL
12 PHP配置数据库MySQL
路老师分享了PHP操作MySQL数据库的方法,包括安装并连接MySQL服务器、选择数据库、执行SQL语句(如插入、更新、删除和查询),以及将结果集返回到数组。通过具体示例代码,详细介绍了每一步的操作流程,帮助读者快速入门PHP与MySQL的交互。
13 1