最近有客户自行升级了系统的JDK版本,原本是1.6的,升到了1.8。导致出现了一个偶发的bug
Comparison method violates its general contract
查询JDK的资料得知,在JDK1.7开始对Comparator类进行了优化,必须包含3个特性。比如在1.6的时候可以只返回-1和1,但是在1.7之后某些情况必须返回0。
借用查询的资料,Comparator必须包含:
自反性:当两个相同的元素相比,compare方法必须返回0,也就是compare(o1, o1) = 0;
反对称性:如果compare(o1,o2) = 1,则compare(o2, o1)必须返回符号相反的值也就是 -1;
传递性:如果 a>b, b>c, 则 a必然大于c。也就是compare(a,b)>0, compare(b,c)>0, 则compare(a,c)>0
系统原来的代码是这样:
public int compare(Object o1, Object o2) {
return o1.getScore() > o2.getScore() ? -1 : 1;
}
这样当两个分数score相等时,就没有返回0,违反了自反性,导致Comparison method violates its general contract的报错
优化后改成compareTo方法,是这样的:
public int compare(Object o1, Object o2) {
return -o1.getScore().compareTo(o2.getScore());
}
问题解决。
后面了解到有另外一种可以暂时不修改代码的解决方法,就是在java启动时加上参数
-Djava.util.Arrays.useLegacyMergeSort=true
有用户反馈也能解决,没有出现报错。