Java异常:IllegalArgumentException Collections.sort报错

简介: Java异常:IllegalArgumentException Collections.sort报错

异常

java.lang.IllegalArgumentException: Comparison method violates its general contract!

at java.util.TimSort.mergeHi(Unknown Source)


其中借鉴 https://www.cnblogs.com/firstdream/p/7204067.html

Exception in thread "main" java.lang.IllegalArgumentException: Comparison method violates its general contract!
  at java.util.TimSort.mergeHi(Unknown Source)
  at java.util.TimSort.mergeAt(Unknown Source)
  at java.util.TimSort.mergeForceCollapse(Unknown Source)
  at java.util.TimSort.sort(Unknown Source)
  at java.util.TimSort.sort(Unknown Source)
  at java.util.Arrays.sort(Unknown Source)
  at java.util.Collections.sort(Unknown Source)
  at com.fc.test.sort(test.java:193)
  at com.fc.test.main(test.java:165)

是JDK的原因


原因


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


在 JDK7 版本以上,Comparator 要满足自反性,传递性,对称性,不然 Arrays.sort,


Collections.sort 会报 IllegalArgumentException 异常。


说明:


1) 自反性:x,y 的比较结果和 y,x 的比较结果相反。


2) 传递性:x>y,y>z,则 x>z。


3) 对称性:x=y,则 x,z 比较结果和 y,z 比较结果相同。


反例:下例中没有处理相等的情况,实际使用中可能会出现异常: 不能使用三元运算

new Comparator<Student>() {
    @Override
    public int compare(Student o1, Student o2) {
        return o1.getId() > o2.getId() ? 1 : -1;
    }
} 

解决方案一

Collections.sort(list, new Comparator<Integer>() {  
    @Override  
    public int compare(Integer o1, Integer o2) {  
        // return o1 > o2 ? 1 : -1;  
        return o1.compareTo(o2);// 正确的方式  
    }  
});  

解决方案二

不修改代码

那么问题来了。为什么上面代码在JDK6中运行无问题,而在JDK7中却会抛异常呢?这是因为JDK7底层的排序算法换了,如果要继续使用JDK6的排序算法,可以在JVM的启动参数中加入如下参数:

-Djava.util.Arrays.useLegacyMergeSort=true  

这样就会照旧使用JDK6的排序算法,在不能修改代码的情况下,解决这个兼容的问题。

目录
相关文章
|
19小时前
|
Java 机器人 程序员
如何解决Java中的ClassCastException异常?
如何解决Java中的ClassCastException异常?
|
19小时前
|
Java 机器人 程序员
Java中如何处理ArrayIndexOutOfBoundsException异常?
Java中如何处理ArrayIndexOutOfBoundsException异常?
|
19小时前
|
安全 Java 机器人
解决Java中的NumberFormatException异常的实用方法
解决Java中的NumberFormatException异常的实用方法
|
19小时前
|
安全 Java 机器人
如何在Java中优雅地处理ConcurrentModificationException异常?
如何在Java中优雅地处理ConcurrentModificationException异常?
|
19小时前
|
Java 机器人 编译器
解决Java中的NoSuchElementException异常的常见方法
解决Java中的NoSuchElementException异常的常见方法
|
19小时前
|
Java 机器人 程序员
Java中如何处理FileNotFoundException异常?
Java中如何处理FileNotFoundException异常?
|
19小时前
|
Java 机器人 程序员
如何解决Java中的IOException异常?
如何解决Java中的IOException异常?
|
19小时前
|
SQL Java 机器人
Java中如何处理SQLException异常?
Java中如何处理SQLException异常?
|
19小时前
|
缓存 监控 Java
解决Java中的OutOfMemoryError异常的有效策略
解决Java中的OutOfMemoryError异常的有效策略
|
19小时前
|
算法 Java 机器人
如何在Java中处理StackOverflowError异常?
如何在Java中处理StackOverflowError异常?