在Java的Stream流操作中,对包含null元素的流进行排序或者获取最值是非常常见的需求,直接使用默认的排序规则很容易触发NullPointerException,而结合Stream.max()和Comparator.nullsLast()可以优雅解决这个问题。

核心API介绍
Stream.max()方法
Stream.max()是Stream接口提供的终端操作,接收一个Comparator作为参数,返回流中的最大元素,结果用Optional包装,避免直接返回null导致的空指针问题。如果流为空,会返回空的Optional对象。
Comparator.nullsLast()方法
Comparator.nullsLast()是一个静态方法,接收一个Comparator作为参数,返回一个可以处理空值的比较器。它会把所有的null值放到非空元素的最后面,非空元素之间的比较使用传入的比较器规则。对应的还有Comparator.nullsFirst(),会把null值放到最前面。
基础使用场景
假设我们有一个包含null的整数列表,需要获取其中的最大元素,空值需要排在最后,不参与最大值的比较,就可以用如下方式实现。
import java.util.Arrays;
import java.util.Comparator;
import java.util.List;
import java.util.Optional;
public class StreamMaxDemo {
public static void main(String[] args) {
// 包含空值的整数列表
List<Integer> numList = Arrays.asList(3, 5, null, 1, 8, null, 2);
// 结合nullsLast处理空值,获取最大值
Optional<Integer> maxNum = numList.stream()
.max(Comparator.nullsLast(Integer::compareTo));
// 输出结果,这里会输出8
maxNum.ifPresent(System.out::println);
}
}
上面的代码中,Comparator.nullsLast(Integer::compareTo)会先判断元素是否为null,把所有null放到最后,然后非空元素按照Integer的自然顺序比较大小,Stream.max()最终会返回非空元素中的最大值8。
自定义对象的场景
如果是自定义对象,需要按照某个属性排序获取最大值,同样可以结合这两个方法使用。比如我们有一个User类,包含age属性,部分User对象的age可能为null,需要获取年龄最大的User,空年龄的User排在最后。
import java.util.Arrays;
import java.util.Comparator;
import java.util.List;
import java.util.Optional;
class User {
private String name;
private Integer age;
public User(String name, Integer age) {
this.name = name;
this.age = age;
}
public Integer getAge() {
return age;
}
@Override
public String toString() {
return "User{name='" + name + "', age=" + age + "}";
}
}
public class CustomObjectDemo {
public static void main(String[] args) {
List<User> userList = Arrays.asList(
new User("张三", 20),
new User("李四", null),
new User("王五", 25),
new User("赵六", 18),
new User("孙七", null)
);
// 按照age属性排序,空age放到最后,获取age最大的用户
Optional<User> maxAgeUser = userList.stream()
.max(Comparator.nullsLast(
Comparator.comparing(User::getAge, Comparator.nullsLast(Integer::compareTo))
));
// 输出结果,这里会输出User{name='王五', age=25}
maxAgeUser.ifPresent(System.out::println);
}
}
这里需要注意,如果自定义对象的属性本身也可能为null,需要在comparing的时候再套一层nullsLast处理属性级别的空值,避免属性比较时抛出空指针。
常见注意事项
- Stream.max()返回的是Optional对象,需要调用isPresent()判断是否有值,或者用ifPresent()处理,避免直接调用get()导致NoSuchElementException。
- Comparator.nullsLast()的参数是不能为null的,如果传入null会直接抛出空指针异常,所以需要保证传入的比较器是有效的。
- 如果需要把空值放到最前面,只需要把nullsLast换成nullsFirst即可,其他逻辑不需要修改。
- 这种方法不仅适用于max()操作,也适用于sorted()等需要比较器的流操作,处理逻辑是一致的。
对比传统处理方式
如果不使用这两个API,传统处理空值排序的方式是先过滤掉null元素,再排序获取最大值,代码如下:
import java.util.Arrays;
import java.util.List;
import java.util.Optional;
public class TraditionalDemo {
public static void main(String[] args) {
List<Integer> numList = Arrays.asList(3, 5, null, 1, 8, null, 2);
// 传统方式:先过滤null再求最大值
Optional<Integer> maxNum = numList.stream()
.filter(num -> num != null)
.max(Integer::compareTo);
maxNum.ifPresent(System.out::println);
}
}
这种方式虽然也能实现需求,但是如果需要保留空值的位置(比如排序后空值在最后),传统方式就无法满足,而使用Comparator.nullsLast()既可以实现空值的安全处理,也能满足空值位置的要求,灵活性更高。
Stream_maxComparator_nullsLast流排序空值处理Java修改时间:2026-06-30 13:39:37