comparator接口用于定义对象的外部自定义比较逻辑,支持lambda、链式多条件排序(comparing/thencomparing)、静态复用及与comparable协同使用。

Java中的Comparator接口用于定义对象之间的自定义比较逻辑,是实现灵活排序的核心工具。它不依赖于类本身的实现(不像Comparable),而是通过外部“比较器”来决定两个对象谁大谁小,特别适合多维度、临时性或第三方类的排序需求。
Comparator的基本用法:Lambda与匿名类
最常用的方式是配合Arrays.sort()或Collection.sort()使用。例如对String列表按长度排序:
✅ 推荐写法(Lambda):
List<String> list = Arrays.asList("apple", "cat", "dog", "banana");
list.sort((s1, s2) -> Integer.compare(s1.length(), s2.length()));✅ 等效写法(匿名类):
立即学习“Java免费学习笔记(深入)”;
list.sort(new Comparator<String>() {
public int compare(String s1, String s2) {
return Integer.compare(s1.length(), s2.length());
}
});注意:compare()方法必须返回负数、0或正数,分别表示“小于”、“等于”、“大于”。优先用Integer.compare(a, b)而非a - b,避免整数溢出。
链式比较与多条件排序
当需要按多个字段排序(如先按年龄升序,年龄相同时按姓名降序),可使用Comparator.comparing()配合thenComparing():
List<Person> people = ...;
people.sort(
Comparator.comparing(Person::getAge)
.thenComparing(Person::getName, Comparator.reverseOrder())
);常见组合方式:
-
comparing(Function):主排序依据 -
thenComparing(Function):次要升序依据 -
thenComparing(Comparator):自定义次要比较器(如反向) -
reversed():整体反转顺序
静态工具方法与复用性设计
为提升可读性和复用性,建议将常用比较器声明为public static final字段:
public class Person {
public static final Comparator<Person> BY_AGE_THEN_NAME =
Comparator.comparing(Person::getAge)
.thenComparing(Person::getName);
<pre class="brush:php;toolbar:false;"><code>// ...}
这样可在任意位置直接使用:people.sort(Person.BY_AGE_THEN_NAME)。也支持提取为独立类(如NameComparator),便于单元测试和注入。
与Comparable的区别与协作
Comparable定义的是类的“自然顺序”,一个类只能有一个;Comparator是“外部策略”,可无限扩展。二者可协同使用:
- 若类实现了
Comparable,调用sort(list)会默认使用它 - 显式传入
Comparator时,它会覆盖自然顺序 - 在
Comparator中可调用compareTo()(如o1.compareTo(o2))复用已有逻辑
典型场景:String本身有自然顺序(字典序),但你可能需要按长度、忽略大小写或倒序排列——这时Comparator就是唯一选择。










