Comparator接口用于定义外部排序规则,通过compare方法返回负数、0或正数决定元素顺序,支持匿名类、Lambda和方法引用三种写法,并可传入Collections.sort、Arrays.sort或TreeSet等实现灵活多条件排序。

Java 里的 Comparator 接口是用来定义**外部排序规则**的核心工具,不改动类本身就能灵活控制排序逻辑。它和 Comparable(类内自然排序)不同,Comparator 是“插拔式”的——你想按什么排、什么时候排,全由你临时决定。
Comparator 的核心规则
它只有一个必须实现的方法:int compare(T o1, T o2)。返回值含义很直接:
- 返回 负数:o1 应排在 o2 前面(比如升序时 o1 更小)
- 返回 0:o1 和 o2 被视为相等,顺序通常保持稳定(不强制交换)
- 返回 正数:o1 应排在 o2 后面(比如升序时 o1 更大)
⚠️ 注意:别直接用 a - b 比较整数,可能溢出。推荐用 Integer.compare(a, b)、Double.compare(a, b) 等安全方法。
三种常用写法(从老到新)
你可以按项目 JDK 版本和可读性需要选择:
立即学习“Java免费学习笔记(深入)”;
-
匿名内部类(兼容老版本,逻辑清晰):
new Comparator() { public int compare(String s1, String s2) { return Integer.compare(s1.length(), s2.length()); } } -
Lambda 表达式(Java 8+ 主流写法,简洁):
(s1, s2) -> Integer.compare(s1.length(), s2.length()) -
方法引用 + 静态工具(JDK 8+ 最简):
Comparator.comparing(String::length)或链式:Comparator.comparing(Student::getAge).thenComparing(Student::getName)
怎么用在实际排序中
Comparator 不自己排序,而是被传给标准排序方法:
- 对 List 排序 → 用
Collections.sort(list, comparator)或list.sort(comparator) - 对 数组 排序 → 用
Arrays.sort(array, comparator) - 构造有序集合 → 如
TreeSet(Comparator.comparing(String::length))
例如:按学生年龄降序、同龄再按姓名升序:
Comparatorcmp = Comparator.comparing(Student::getAge).reversed()
.thenComparing(Student::getName);
自定义类排序的典型步骤
假设你有一个 Book 类,想按页数排序:
- 不用改
Book类(不实现 Comparable) - 写一个比较器:Lambda 就够了 ——
(b1, b2) -> Integer.compare(b1.getPages(), b2.getPages()) - 调用
Collections.sort(books, 上面那个 comparator) - 如果要多条件,直接链式追加:
.thenComparing(Book::getTitle)
整个过程完全解耦,同一个类可以同时支持“按价格排”“按出版年排”“按作者姓氏排”,互不干扰。










