0

0

使用Java Stream Filter实现多个函数式接口过滤

DDD

DDD

发布时间:2025-10-12 11:43:35

|

234人浏览过

|

来源于php中文网

原创

使用java stream filter实现多个函数式接口过滤

本文旨在讲解如何利用Java Stream API和函数式接口,对集合数据进行多重条件过滤。我们将探讨如何有效地组合多个Predicate,实现“与”、“或”、“非”等逻辑运算,并提供多种实现方案,帮助你编写简洁高效的过滤代码。

理解函数式接口 Predicate

在Java中,Predicate是一个函数式接口,它接受一个类型为T的参数,并返回一个布尔值。Predicate接口的核心方法是test(T t),用于判断给定的输入是否满足特定条件。 使用Predicate接口可以方便地定义过滤条件,并将其应用于Stream API的filter操作中。

import java.time.LocalDate;
import java.time.temporal.ChronoUnit;
import java.util.function.Predicate;

import lombok.Getter;

class Student {
  @Getter private LocalDate birthDate;

  public Student(LocalDate birthDate) {
        this.birthDate = birthDate;
  }
}

public class PredicateExample {

    public static void main(String[] args) {
        Predicate<Student> isAdult = s -> ChronoUnit.YEARS.between(
          LocalDate.now(), s.getBirthDate()) >= 18;

        Student student1 = new Student(LocalDate.now().minusYears(20));
        Student student2 = new Student(LocalDate.now().minusYears(16));

        System.out.println("Student 1 is adult: " + isAdult.test(student1)); // Output: true
        System.out.println("Student 2 is adult: " + isAdult.test(student2)); // Output: false
    }
}

组合多个 Predicate

当需要应用多个过滤条件时,可以将多个Predicate组合起来。Predicate接口提供了and、or和negate等方法,用于实现逻辑与、逻辑或和逻辑非操作。

1. 逻辑与 (AND)

要实现所有Predicate都满足时才保留元素,可以使用and方法。

立即学习Java免费学习笔记(深入)”;

Q.AI视频生成工具
Q.AI视频生成工具

支持一分钟生成专业级短视频,多种生成方式,AI视频脚本,在线云编辑,画面自由替换,热门配音媲美真人音色,更多强大功能尽在QAI

下载
import java.time.LocalDate;
import java.time.temporal.ChronoUnit;
import java.util.Arrays;
import java.util.List;
import java.util.function.Predicate;
import java.util.stream.Collectors;

import lombok.Getter;

class Student {
    @Getter private LocalDate birthDate;
    @Getter private String name;

    public Student(LocalDate birthDate, String name) {
        this.birthDate = birthDate;
        this.name = name;
    }
}

public class AndPredicateExample {

    public static void main(String[] args) {
        List<Student> students = Arrays.asList(
            new Student(LocalDate.now().minusYears(20), "Alice"),
            new Student(LocalDate.now().minusYears(16), "Bob"),
            new Student(LocalDate.now().minusYears(22), "Charlie")
        );

        Predicate<Student> isAdult = s -> ChronoUnit.YEARS.between(LocalDate.now(), s.getBirthDate()) >= 18;
        Predicate<Student> nameStartsWithA = s -> s.getName().startsWith("A");

        // 使用 and 方法组合 Predicate
        Predicate<Student> combinedPredicate = isAdult.and(nameStartsWithA);

        // 使用 Stream API 进行过滤
        List<Student> filteredStudents = students.stream()
            .filter(combinedPredicate)
            .collect(Collectors.toList());

        filteredStudents.forEach(student -> System.out.println(student.getName())); // Output: Alice
    }
}

2. 逻辑或 (OR)

要实现只要有一个Predicate满足就保留元素,可以使用or方法。

import java.time.LocalDate;
import java.time.temporal.ChronoUnit;
import java.util.Arrays;
import java.util.List;
import java.util.function.Predicate;
import java.util.stream.Collectors;

import lombok.Getter;

class Student {
    @Getter private LocalDate birthDate;
    @Getter private String name;

    public Student(LocalDate birthDate, String name) {
        this.birthDate = birthDate;
        this.name = name;
    }
}

public class OrPredicateExample {

    public static void main(String[] args) {
        List<Student> students = Arrays.asList(
            new Student(LocalDate.now().minusYears(20), "Alice"),
            new Student(LocalDate.now().minusYears(16), "Bob"),
            new Student(LocalDate.now().minusYears(22), "Charlie")
        );

        Predicate<Student> isAdult = s -> ChronoUnit.YEARS.between(LocalDate.now(), s.getBirthDate()) >= 18;
        Predicate<Student> nameStartsWithB = s -> s.getName().startsWith("B");

        // 使用 or 方法组合 Predicate
        Predicate<Student> combinedPredicate = isAdult.or(nameStartsWithB);

        // 使用 Stream API 进行过滤
        List<Student> filteredStudents = students.stream()
            .filter(combinedPredicate)
            .collect(Collectors.toList());

        filteredStudents.forEach(student -> System.out.println(student.getName())); // Output: Alice, Bob, Charlie
    }
}

3. 逻辑非 (NOT)

要实现对Predicate取反,可以使用negate方法。

import java.time.LocalDate;
import java.time.temporal.ChronoUnit;
import java.util.Arrays;
import java.util.List;
import java.util.function.Predicate;
import java.util.stream.Collectors;

import lombok.Getter;

class Student {
    @Getter private LocalDate birthDate;
    @Getter private String name;

    public Student(LocalDate birthDate, String name) {
        this.birthDate = birthDate;
        this.name = name;
    }
}

public class NegatePredicateExample {

    public static void main(String[] args) {
        List<Student> students = Arrays.asList(
            new Student(LocalDate.now().minusYears(20), "Alice"),
            new Student(LocalDate.now().minusYears(16), "Bob"),
            new Student(LocalDate.now().minusYears(22), "Charlie")
        );

        Predicate<Student> isAdult = s -> ChronoUnit.YEARS.between(LocalDate.now(), s.getBirthDate()) >= 18;

        // 使用 negate 方法取反 Predicate
        Predicate<Student> combinedPredicate = isAdult.negate();

        // 使用 Stream API 进行过滤
        List<Student> filteredStudents = students.stream()
            .filter(combinedPredicate)
            .collect(Collectors.toList());

        filteredStudents.forEach(student -> System.out.println(student.getName())); // Output: Bob
    }
}

使用循环组合 Predicate

如果需要动态地组合多个Predicate,可以使用循环来实现。例如,将多个Predicate进行逻辑与操作:

import java.time.LocalDate;
import java.time.temporal.ChronoUnit;
import java.util.Arrays;
import java.util.List;
import java.util.function.Predicate;
import java.util.stream.Collectors;

import lombok.Getter;

class Student {
    @Getter private LocalDate birthDate;
    @Getter private String name;

    public Student(LocalDate birthDate, String name) {
        this.birthDate = birthDate;
        this.name = name;
    }
}

public class LoopPredicateExample {

    public static void main(String[] args) {
        List<Student> students = Arrays.asList(
            new Student(LocalDate.now().minusYears(20), "Alice"),
            new Student(LocalDate.now().minusYears(16), "Bob"),
            new Student(LocalDate.now().minusYears(22), "Charlie")
        );

        List<pre class="brush:php;toolbar:false;"dicate<Student>> filters = Arrays.asList(
            s -> ChronoUnit.YEARS.between(LocalDate.now(), s.getBirthDate()) >= 18,
            s -> s.getName().startsWith("A")
        );

        // 使用循环组合 Predicate
        Predicate<Student> combinedPredicate = s -> true;
        for (Predicate<Student> filter : filters) {
            combinedPredicate = combinedPredicate.and(filter);
        }

        // 使用 Stream API 进行过滤
        List<Student> filteredStudents = students.stream()
            .filter(combinedPredicate)
            .collect(Collectors.toList());

        filteredStudents.forEach(student -> System.out.println(student.getName())); // Output: Alice
    }
}

使用 Stream API 的 reduce 操作组合 Predicate

除了循环,还可以使用Stream API的reduce操作来组合Predicate。

import java.time.LocalDate;
import java.time.temporal.ChronoUnit;
import java.util.Arrays;
import java.util.List;
import java.util.function.Predicate;
import java.util.stream.Collectors;

import lombok.Getter;

class Student {
    @Getter private LocalDate birthDate;
    @Getter private String name;

    public Student(LocalDate birthDate, String name) {
        this.birthDate = birthDate;
        this.name = name;
    }
}

public class ReducePredicateExample {

    public static void main(String[] args) {
        List<Student> students = Arrays.asList(
            new Student(LocalDate.now().minusYears(20), "Alice"),
            new Student(LocalDate.now().minusYears(16), "Bob"),
            new Student(LocalDate.now().minusYears(22), "Charlie")
        );

        List<pre class="brush:php;toolbar:false;"dicate<Student>> filters = Arrays.asList(
            s -> ChronoUnit.YEARS.between(LocalDate.now(), s.getBirthDate()) >= 18,
            s -> s.getName().startsWith("A")
        );

        // 使用 Stream API 的 reduce 操作组合 Predicate
        Predicate<Student> combinedPredicate = filters.stream()
            .reduce(s -> true, Predicate::and);

        // 使用 Stream API 进行过滤
        List<Student> filteredStudents = students.stream()
            .filter(combinedPredicate)
            .collect(Collectors.toList());

        filteredStudents.forEach(student -> System.out.println(student.getName())); // Output: Alice
    }
}

注意事项

  • 确保Predicate的逻辑清晰,避免出现复杂的嵌套条件。
  • 尽量使用Predicate接口提供的and、or和negate方法,提高代码的可读性。
  • 在处理大量数据时,注意Predicate的性能,避免出现性能瓶颈

总结

本文介绍了如何使用Java Stream API和Predicate接口,对集合数据进行多重条件过滤。通过组合多个Predicate,可以实现复杂的过滤逻辑,提高代码的灵活性和可维护性。掌握这些技巧,可以编写出更加简洁高效的过滤代码。

热门AI工具

更多
DeepSeek
DeepSeek

幻方量化公司旗下的开源大模型平台

豆包大模型
豆包大模型

字节跳动自主研发的一系列大型语言模型

通义千问
通义千问

阿里巴巴推出的全能AI助手

腾讯元宝
腾讯元宝

腾讯混元平台推出的AI助手

文心一言
文心一言

文心一言是百度开发的AI聊天机器人,通过对话可以生成各种形式的内容。

讯飞写作
讯飞写作

基于讯飞星火大模型的AI写作工具,可以快速生成新闻稿件、品宣文案、工作总结、心得体会等各种文文稿

即梦AI
即梦AI

一站式AI创作平台,免费AI图片和视频生成。

ChatGPT
ChatGPT

最最强大的AI聊天机器人程序,ChatGPT不单是聊天机器人,还能进行撰写邮件、视频脚本、文案、翻译、代码等任务。

相关专题

更多
硬盘接口类型介绍
硬盘接口类型介绍

硬盘接口类型有IDE、SATA、SCSI、Fibre Channel、USB、eSATA、mSATA、PCIe等等。详细介绍:1、IDE接口是一种并行接口,主要用于连接硬盘和光驱等设备,它主要有两种类型:ATA和ATAPI,IDE接口已经逐渐被SATA接口;2、SATA接口是一种串行接口,相较于IDE接口,它具有更高的传输速度、更低的功耗和更小的体积;3、SCSI接口等等。

1902

2023.10.19

PHP接口编写教程
PHP接口编写教程

本专题整合了PHP接口编写教程,阅读专题下面的文章了解更多详细内容。

656

2025.10.17

php8.4实现接口限流的教程
php8.4实现接口限流的教程

PHP8.4本身不内置限流功能,需借助Redis(令牌桶)或Swoole(漏桶)实现;文件锁因I/O瓶颈、无跨机共享、秒级精度等缺陷不适用高并发场景。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。

2387

2025.12.29

java接口相关教程
java接口相关教程

本专题整合了java接口相关内容,阅读专题下面的文章了解更多详细内容。

47

2026.01.19

Go高并发任务调度与Goroutine池化实践
Go高并发任务调度与Goroutine池化实践

本专题围绕 Go 语言在高并发任务处理场景中的实践展开,系统讲解 Goroutine 调度模型、Channel 通信机制以及并发控制策略。内容包括任务队列设计、Goroutine 池化管理、资源限制控制以及并发任务的性能优化方法。通过实际案例演示,帮助开发者构建稳定高效的 Go 并发任务处理系统,提高系统在高负载环境下的处理能力与稳定性。

22

2026.03.10

Kotlin Android模块化架构与组件化开发实践
Kotlin Android模块化架构与组件化开发实践

本专题围绕 Kotlin 在 Android 应用开发中的架构实践展开,重点讲解模块化设计与组件化开发的实现思路。内容包括项目模块拆分策略、公共组件封装、依赖管理优化、路由通信机制以及大型项目的工程化管理方法。通过真实项目案例分析,帮助开发者构建结构清晰、易扩展且维护成本低的 Android 应用架构体系,提升团队协作效率与项目迭代速度。

48

2026.03.09

JavaScript浏览器渲染机制与前端性能优化实践
JavaScript浏览器渲染机制与前端性能优化实践

本专题围绕 JavaScript 在浏览器中的执行与渲染机制展开,系统讲解 DOM 构建、CSSOM 解析、重排与重绘原理,以及关键渲染路径优化方法。内容涵盖事件循环机制、异步任务调度、资源加载优化、代码拆分与懒加载等性能优化策略。通过真实前端项目案例,帮助开发者理解浏览器底层工作原理,并掌握提升网页加载速度与交互体验的实用技巧。

93

2026.03.06

Rust内存安全机制与所有权模型深度实践
Rust内存安全机制与所有权模型深度实践

本专题围绕 Rust 语言核心特性展开,深入讲解所有权机制、借用规则、生命周期管理以及智能指针等关键概念。通过系统级开发案例,分析内存安全保障原理与零成本抽象优势,并结合并发场景讲解 Send 与 Sync 特性实现机制。帮助开发者真正理解 Rust 的设计哲学,掌握在高性能与安全性并重场景中的工程实践能力。

216

2026.03.05

PHP高性能API设计与Laravel服务架构实践
PHP高性能API设计与Laravel服务架构实践

本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。

412

2026.03.04

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
Kotlin 教程
Kotlin 教程

共23课时 | 4.3万人学习

C# 教程
C# 教程

共94课时 | 11.1万人学习

Java 教程
Java 教程

共578课时 | 80.6万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号