排序是编程中经常需要用到的功能,而 Java 的工具类 Collections 中也提供了 sort 方法用于实现对列表等集合中元素的排序。Collections.sort() 方法有两种形式:Collections.sort(List)Collections.sort(List, Comparator)

  第一种 Collections.sort(List) 要求 List 中的元素已经实现了 Comparable 接口,第二种 Collections.sort(List, Comparator) 则需要实现一个比较器。基于比较的排序的关键就是能比较两个元素的大小,知道了谁大谁小,才能确定谁排在前面谁排在后面;而这两种方法的本质都是为了确定两个元素谁大谁小。接下来主要谈一谈基于比较器的自定义排序。

  先看个排序示例:

public static void main(String[] args) {
	Integer[] input = { 1, 3, 8, 7, 4, 9, 2 };        
	List<Integer> lst = Arrays.asList(input);
	
	Collections.sort(lst);
	System.out.println(lst);
}
//输出为: [1, 2, 3, 4, 7, 8, 9]

  可以看到,原本无序的列表变成了从小到大排列。原因有二:

  1. Integer 类实现了 Comparable 接口,使得元素之间可以比较大小
  2. sort 方法默认是从小到大排列(这点非常重要)

  然而大多数情况下,我们想要的都不是简单的升序排列,这时候自定义排序就要派上用场了。Comparator 实现自定义排序的关键是实现 compare 方法,该方法声明为:

int compare(T o1, T o2);

当方法返回负值时,代表 o1 < o2;当方法返回 0 时,代表 o1 = o2;当方法返回正值时,代表 o1 > o2;下面以实现降序排列为例,演示如何使用 Comparator。

###一、降序排列

public class Main {
    public static void main(String[] args) {
        Integer[] input = { 1, 3, 8, 7, 4, 9, 2 };        
        List<Integer> lst = Arrays.asList(input);
        
        Collections.sort(lst, new IntComparator<Integer>());
        System.out.println(lst);
    }
}

class IntComparator<T> implements Comparator<T> {
    public int compare(T a, T b) {
        return (Integer)b - (Integer)a;
    }
}
//输出为: [9, 8, 7, 4, 3, 2, 1]

  为什么这样就实现了降序排列呢?

  这是因为当传入 Comparator 参数时, sort 方法会将需要比较大小的两个元素 a,b 作为参数传给 Comparator.compare(T a, T b) 方法,根据返回的结果判断 a,b 的大小。例如,传入1 和 3 的时候,compare 返回 2,正值代表 a > b,即 1 “大于” 3,这个“大于”正是我们自己定义的规则。然后,根据 sort 方法默认从小到大排序的性质,小的排在前面,所以最后 3 排在 1 前面,也就实现了降序排列。

  在这里 return (Integer)b - (Integer)a; 是一种经验写法,如果分解开来应该是这样:

class IntComparator<T> implements Comparator<T> {
    public int compare(T a, T b) {
        int num1 = (Integer)a;
        int num2 = (Integer)b;
        
        if(num1 > num2) { //如果 a>b,根据自定义规则 a 应该在前面,即自定义规则中 a “小于” b,故此时返回负值
            return -1;
        } else if(num1 < num2) {
            return 1;
        } else {
            return 0;
        }
    }
}

  该 compare 方法非常简单,但在实际应用中,由于自定义规则的复杂性,compare 的编写难度也可能变得很大。下面再举一例,让大家加深对 compare 的理解。

###二、奇偶排列   要求:奇数排在偶数前面,奇数之间从小到大排列,偶数之间从大到小排序

class IntComparator<T> implements Comparator<T> {
    public int compare(T a, T b) {
        int num1 = (Integer)a;
        int num2 = (Integer)b;
        if(num1 % 2 == 1 && num2 % 2 == 1) {
            return num1 - num2; // ①
        } else if(num1 % 2 == 0 && num2 % 2 == 0) {
            return num2 - num1; // ②
        } else if(num1 % 2 == 1 && num2 % 2 == 0) {
            return -1; // ③
        } else {
            return 1; // ④
        }
    }
}

  ① ② 表示两数同奇或同偶的情况,此时根据经验写法很容易得到。

  ③ ④ 表示奇偶不同的情况,此时若 a 为奇数,b 为偶数,则 a 应该排在 b 前面,即 a “小于” b,故返回负值;反之,返回正值。

  最后,再次强调:排在前面的元素“小”,排在后面的元素“大”

  留个练习,有兴趣的朋友可以做下,看看自己是不是真的掌握了。

  练习题:实现数字按照高位到低位的比较,进行降序排序;高位相同的,位数少的排前面。例如:输入 {3, 34, 5, 7, 56, 523, 52, 571};输出 {7, 5, 571, 56, 52, 523, 3, 34}