Java中List排序核心是Collections.sort()或List.sort()配合Comparable/Comparator,推荐用Comparator自定义规则,注意null处理、不可变列表转换、线程安全及比较器合规性。

Java中为List排序,核心是利用Collections.sort()或List.sort()方法,配合Comparable接口或Comparator实现。关键不在于“能不能排”,而在于“按什么规则、怎么写得清晰安全、怎么避免常见坑”。
用Comparator自定义排序逻辑(最常用)
当List元素类型未实现Comparable,或需多种排序方式(如按姓名升序、按年龄降序),推荐使用Comparator。Java 8+ 支持Lambda和方法引用,写法简洁且不易出错。
- 按字符串长度升序:
list.sort(Comparator.comparing(String::length)); - 按年龄降序(假设对象有getAge()):
list.sort(Comparator.comparing(Person::getAge).reversed()); - 多字段组合:先按部门升序,部门相同时按薪资降序:
list.sort(Comparator.comparing(Person::getDept).thenComparing(Person::getSalary, Comparator.reverseOrder())); - 注意null值:若字段可能为null,用
Comparator.nullsLast()或nullsFirst()包裹,避免NullPointerException,例如:Comparator.nullsLast(Comparator.naturalOrder())
让元素自身支持自然排序(实现Comparable)
如果该类“默认就该按某规则排序”(如Person默认按ID排序),可让其实现Comparable接口。这样调用Collections.sort(list)无需额外参数,语义更明确。
- 在类中重写
compareTo(),返回负数、0、正数表示小于、等于、大于 - 推荐用
Integer.compare(a, b)、Objects.compare(a, b, comparator)等工具方法,避免手动减法导致的整型溢出或空指针 - 若已有继承关系或无法修改源码,此方式不适用,优先选
Comparator
不可变List与并发场景下的排序处理
直接对Arrays.asList()或List.of()返回的List调用sort()会抛UnsupportedOperationException——它们是不可修改的视图或不可变实例。
立即学习“Java免费学习笔记(深入)”;
- 先转为可变ArrayList:
new ArrayList(originalList).sort(comparator); - 并发环境下,
ArrayList本身不保证线程安全。若需多线程排序,应确保操作时无其他线程修改该List;更稳妥做法是复制一份再排序:Listsorted = new ArrayList(list); sorted.sort(comparator); - Stream排序(Java 8+)天然无副作用:
List,适合函数式风格或需链式处理的场景sorted = list.stream().sorted(comparator).collect(Collectors.toList());
避开常见陷阱
排序看似简单,但几个细节常被忽略,导致结果异常或运行时报错。
-
比较器必须满足“自反性、对称性、传递性”,否则
sort()行为未定义(可能抛IllegalArgumentException或死循环)。例如,用return a > b ? 1 : -1;忽略相等情况,违反自反性 -
基本类型包装类排序慎用减法:
Integer a = 2000000000, b = -2000000000; return a - b;会溢出成负数,误判大小。统一用Integer.compare(a, b) -
日期排序别用getTime()硬转long,优先用
LocalDateTime.compareTo()或Comparator.comparing(Obj::getInstant),语义更清晰且时区安全 - 排序前确认List非null,否则直接NPE;若允许null元素,务必在Comparator中显式处理
基本上就这些。掌握Comparator的灵活构造、理解可变性约束、绕开比较逻辑的坑,List排序就稳了。










