Predicate和Consumer接口– Java 8中java.util.function包下的接口

简介:

早先我写了一篇《函数式接口》,探讨了部分Java 8中函数式接口的用法。我也提及了Predicate接口属于java.util.function包,在这篇文章中,我将展示如何应用Predicate接口和Consumer接口。

一起看一下Predicate的官方文档:

Determines if the input object matches some criteria.

即判断输入的对象是否符合某个条件。

在Predicate接口中,有以下5个方法(你肯定好奇为何此接口属于函数式接口。如果你这么想,在使用接口前应该好好研读方法的注释):

01 //Returns a predicate which evaluates to true only if this predicate
02 //and the provided predicate both evaluate to true.
03 and(Predicate<? super T> p) 
04  
05 //Returns a predicate which negates the result of this predicate.
06 negate() 
07  
08 //Returns a predicate which evaluates to true if either
09 //this predicate or the provided predicate evaluates to true
10 or(Predicate<? super T> p) 
11  
12 //Returns true if the input object matches some criteria
13 test(T t) 
14  
15 //Returns a predicate that evaluates to true if both or neither
16 //of the component predicates evaluate to true
17 xor(Predicate<? super T> p)

除了test()方法是抽象方法以外,其他方法都是默认方法(译者注:在Java 8中,接口可以包含带有实现代码的方法,这些方法称为default方法)。可以使用匿名内部类提供test()方法的实现,也可以使用lambda表达式实现test()。

Consumer接口的文档声明如下:

An operation which accepts a single input argument and returns no result. Unlike most other functional interfaces, Consumer is expected to operate via side-effects.

即接口表示一个接受单个输入参数并且没有返回值的操作。不像其他函数式接口,Consumer接口期望执行带有副作用的操作(译者注:Consumer的操作可能会更改输入参数的内部状态)。

Consumer接口中有2个方法,有且只有一个声明为accept(T t)的方法,接收一个输入参数并且没有返回值。为了详细说明Predicate和Consumer接口,我们来考虑一下学生的例子:Student类包含姓名,分数以及待付费用,每个学生可根据分数获得不同程度的费用折扣。

01 class Student{
02  
03     String firstName;
04  
05     String lastName;
06  
07     Double grade;
08  
09     Double feeDiscount = 0.0;
10  
11     Double baseFee = 20000.0;
12  
13     public Student(String firstName, String lastName, Double grade) {
14  
15         this.firstName = firstName;
16  
17         this.lastName = lastName;
18  
19         this.grade = grade;
20     }
21  
22     public void printFee(){
23  
24         Double newFee = baseFee - ((baseFee * feeDiscount) / 100);
25  
26         System.out.println("The fee after discount: " + newFee);
27  
28     }
29  
30 }

我们分别声明一个接受Student对象的Predicate接口以及Consumer接口的实现类。如果你还不熟悉Function接口,那么你需要花几分钟阅读一下这篇文章。这个例子使用Predicate接口实现类的test()方法判断输入的Student对象是否拥有费用打折的资格,然后使用Consumer接口的实现类更新输入的Student对象的折扣。

01 public class PreidcateConsumerDemo {
02  
03    public static Student updateStudentFee(Student student, Predicate<Student> predicate, Consumer<Student> consumer){
04  
05         //Use the predicate to decide when to update the discount.
06  
07         if ( predicate.test(student)){
08  
09             //Use the consumer to update the discount value.
10  
11             consumer.accept(student);
12         }
13  
14         return student;
15  
16     }
17  
18 }

Predicate和Consumer接口的test()和accept()方法都接受一个泛型参数。不同的是test()方法进行某些逻辑判断并返回一个boolean值,而accept()接受并改变某个对象的内部值。updateStudentFee方法的调用如下所示:

01 public static void main(String[] args) {
02  
03     Student student1 = new Student("Ashok","Kumar"9.5);
04  
05     student1 = updateStudentFee(student1,
06                                 //Lambda expression for Predicate interface
07                                 student -> student.grade > 8.5,
08                                 //Lambda expression for Consumer inerface
09                                 student -> student.feeDiscount = 30.0);
10  
11     student1.printFee();
12  
13     Student student2 = new Student("Rajat","Verma"8.0);
14  
15     student2 = updateStudentFee(student2,
16                                 student -> student.grade >= 8,
17                                 student -> student.feeDiscount = 20.0);
18  
19     student2.printFee();
20  
21 }
相关文章
|
5天前
|
Java 开发者
探索 Java 的函数式接口和 Lambda 表达式
【4月更文挑战第19天】Java 中的函数式接口和 Lambda 表达式提供了简洁、灵活的编程方式。函数式接口有且仅有一个抽象方法,用于与 Lambda(一种匿名函数语法)配合,简化代码并增强可读性。Lambda 表达式的优点在于其简洁性和灵活性,常用于事件处理、过滤和排序等场景。使用时注意兼容性和变量作用域,它们能提高代码效率和可维护性。
|
6天前
|
Java
Java接口中可以定义哪些方法?
【4月更文挑战第13天】
7 0
Java接口中可以定义哪些方法?
|
8天前
|
设计模式 Java
Java接口与抽象类
Java接口与抽象类
17 0
|
12天前
|
安全 Java 编译器
接口之美,内部之妙:深入解析Java的接口与内部类
接口之美,内部之妙:深入解析Java的接口与内部类
34 0
接口之美,内部之妙:深入解析Java的接口与内部类
|
14天前
|
存储 Java
java接口和内部类
java接口和内部类
|
1天前
|
安全 Java 调度
Java线程:深入理解与实战应用
Java线程:深入理解与实战应用
9 0
|
1天前
|
Java
Java中的并发编程:理解和应用线程池
【4月更文挑战第23天】在现代的Java应用程序中,性能和资源的有效利用已经成为了一个重要的考量因素。并发编程是提高应用程序性能的关键手段之一,而线程池则是实现高效并发的重要工具。本文将深入探讨Java中的线程池,包括其基本原理、优势、以及如何在实际开发中有效地使用线程池。我们将通过实例和代码片段,帮助读者理解线程池的概念,并学习如何在Java应用中合理地使用线程池。
|
5天前
|
安全 Java
深入理解 Java 多线程和并发工具类
【4月更文挑战第19天】本文探讨了Java多线程和并发工具类在实现高性能应用程序中的关键作用。通过继承`Thread`或实现`Runnable`创建线程,利用`Executors`管理线程池,以及使用`Semaphore`、`CountDownLatch`和`CyclicBarrier`进行线程同步。保证线程安全、实现线程协作和性能调优(如设置线程池大小、避免不必要同步)是重要环节。理解并恰当运用这些工具能提升程序效率和可靠性。
|
6天前
|
安全 Java
java多线程(一)(火车售票)
java多线程(一)(火车售票)
|
6天前
|
安全 Java 调度
Java并发编程:深入理解线程与锁
【4月更文挑战第18天】本文探讨了Java中的线程和锁机制,包括线程的创建(通过Thread类、Runnable接口或Callable/Future)及其生命周期。Java提供多种锁机制,如`synchronized`关键字、ReentrantLock和ReadWriteLock,以确保并发访问共享资源的安全。此外,文章还介绍了高级并发工具,如Semaphore(控制并发线程数)、CountDownLatch(线程间等待)和CyclicBarrier(同步多个线程)。掌握这些知识对于编写高效、正确的并发程序至关重要。