Java里Stream的filter和map如何使用_Java流式处理说明

filter用于筛选符合条件的元素并生成新流,不修改原数据;map用于逐个转换元素且数量不变;二者顺序影响性能与语义,通常先filter再map更优。

filter 用来保留符合条件的元素,不是“过滤掉”

很多人初看 filter 名字,以为它像数据库 DELETE 那样“删掉”不满足条件的项。其实它只是从原流中**筛选出匹配谓词(Predicate)的元素**,生成新流,原数据不动。

常见错误是写反判断逻辑,比如想取偶数却写了 x % 2 == 1;或对 null 元素没做防护,导致 NullPointerException

  • 务必确保 Predicate 内部不抛未检查异常,否则流会中断
  • 如果源是 Collectionfilter 不改变其内容,只影响后续操作
  • 链式调用中,filter 尽量前置——减少后续 mapreduce 处理的数据量
List numbers = Arrays.asList(1, 2, 3, 4, 5);
List evens = numbers.stream()
    .filter(x -> x != null && x % 2 == 0) // 防 null + 判偶
    .collect(Collectors.toList());

map 负责逐个转换,类型可以变但数量不变

map 接收 Function,对每个元素执行一次转换,输出流的元素个数与输入流严格一致。它不筛选、不聚合、不跳过——哪怕你返回 null,那个 null 也会进结果流(除非后续有 filter(Objects::nonNull))。

典型陷阱:把 map 当成 flatMap 用,比如想把 List 拆成所有字符,却写了 map(s -> s.chars().boxed().collect(...)),结果得到 List> 而非扁平的 List

  • 转换函数里避免副作用(如修改外部变量、IO),流可能并行执行
  • 若转换后需判空或进一步处理,优先在 map 后接 filter,而不是在 map 里塞三元运算返回空集合
  • 注意泛型擦除:从 String 映射到 LocalDate 时,别漏了 DateTimeParseException 的捕获(建议提前 validate 或用 Optional 包装)
List dates = Arrays.asList("2025-01-01", "2025-02-30");
List parsed = dates.stream()
    .map(s -> {
        try {
            return LocalDate.parse(s);
        } catch (DateTimeParseException e) {
            return null; // 后续 filter 可剔除
        }
    })
    .filter(Objects::nonNull)
    .collect(Collectors.toList());

filter 和 map 的顺序会影响性能和语义

filtermap 通常是更优选择,尤其当 map 操作开销大(如解析 JSON、查库、IO)时。反过来不仅浪费计算,还可能因无效输入触发异常。

但也有例外:比如你需要基于转换后的值做判断(如 “找出所有转成整数后大于 100 的字符串”),那就必须先 mapfilter —— 此时建议把解析逻辑封装好,并用 Optional 或异常处理兜底。

  • 不要为了“一行写完”强行嵌套复杂逻辑,可提取为私有方法提升可读性
  • 并行流中,filtermap 都是无状态操作,可安全并行,但顺序仍由链式调用决定
  • 调试时可用 peek 查看中间结果,但上线前务必删掉——它仅用于调试,不保证执行时机

空流、null 元素和提前终止的注意事项

Stream 本身可以为空(如 Stream.empty()),filtermap 都能正常处理;但若流中包含 null 元素,而你的 mapfilter 函数没做防御,就会炸。

另一个易忽略点:findAnyfindFirst 等短路操作,在 filter 后可能提前结束,但 map 总是全量执行(除非上游已短路)。所以别指望 “filter(...).map(...).findFirst()” 会让 map 只跑一次。

  • 对可能含 null 的集合,初始化流前先用 Collection.removeIf(Objects::isNull),或在 filter 中统一拦截
  • map 返回 null 是合法的,但后续收集到 List 没问题,收集到 Map(如用 toMap)就会抛 NullPointerException
  • 需要“找到第一个满足条件并转换”的语义,用 filter(...).map(...).findFirst() 是对的;但若转换成本高,考虑用传统 for 循环手动控制
真正麻烦的不是语法,而是想清楚:你要的是“先筛再算”,还是“先算再筛”,以及算的过程中愿不愿意为少数脏数据多写几行防御代码。