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

jopen 9年前發布 | 21K 次閱讀 Java 8 Java開發

原文鏈接 作者: Mohamed Sanaulla 譯者: 李璟(jlee381344197@gmail.com)

早先我寫了一篇《函數式接口》,探討了部分Java 8中函數式接口的用法。我也提及了Predicate接口屬于java.util.function包,在這篇文章中,我將展示如何應用Predicate接口和Consumer接口。

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

Determines if the input object matches some criteria.

</blockquote>

即判斷輸入的對象是否符合某個條件。

在Predicate接口中,有以下5個方法(你肯定好奇為何此接口屬于函數式接口。如果你這么想,在使用接口前應該好好研讀方法的注釋):

//Returns a predicate which evaluates to true only if this predicate
//and the provided predicate both evaluate to true.
and(Predicate<? super T> p) 

//Returns a predicate which negates the result of this predicate.
negate() 

//Returns a predicate which evaluates to true if either
//this predicate or the provided predicate evaluates to true
or(Predicate<? super T> p) 

//Returns true if the input object matches some criteria
test(T t) 

//Returns a predicate that evaluates to true if both or neither
//of the component predicates evaluate to true
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類包含姓名,分數以及待付費用,每個學生可根據分數獲得不同程度的費用折扣。

class Student{

    String firstName;

    String lastName;

    Double grade;

    Double feeDiscount = 0.0;

    Double baseFee = 20000.0;

    public Student(String firstName, String lastName, Double grade) {

        this.firstName = firstName;

        this.lastName = lastName;

        this.grade = grade;
    }

    public void printFee(){

        Double newFee = baseFee - ((baseFee * feeDiscount) / 100);

        System.out.println("The fee after discount: " + newFee);

    }

}

我們分別聲明一個接受Student對象的Predicate接口以及Consumer接口的實現類。如果你還不熟悉Function接口,那么你需要花幾分鐘閱讀一下這篇文章。這個例子使用Predicate接口實現類的test()方法判斷輸入的Student對象是否擁有費用打折的資格,然后使用Consumer接口的實現類更新輸入的Student對象的折扣。

public class PreidcateConsumerDemo {

   public static Student updateStudentFee(Student student, Predicate<Student> predicate, Consumer<Student> consumer){

        //Use the predicate to decide when to update the discount.

        if ( predicate.test(student)){

            //Use the consumer to update the discount value.

            consumer.accept(student);
        }

        return student;

    }

}

Predicate和Consumer接口的test()和accept()方法都接受一個泛型參數。不同的是test()方法進行某些邏輯判斷并返回一個boolean值,而accept()接受并改變某個對象的內部值。updateStudentFee方法的調用如下所示:

public static void main(String[] args) {

    Student student1 = new Student("Ashok","Kumar", 9.5);

    student1 = updateStudentFee(student1,
                                //Lambda expression for Predicate interface
                                student -> student.grade > 8.5,
                                //Lambda expression for Consumer inerface
                                student -> student.feeDiscount = 30.0);

    student1.printFee();

    Student student2 = new Student("Rajat","Verma", 8.0);

    student2 = updateStudentFee(student2,
                                student -> student.grade >= 8,
                                student -> student.feeDiscount = 20.0);

    student2.printFee();

}
 本文由用戶 jopen 自行上傳分享,僅供網友學習交流。所有權歸原作者,若您的權利被侵害,請聯系管理員。
 轉載本站原創文章,請注明出處,并保留原始鏈接、圖片水印。
 本站是一個以用戶分享為主的開源技術平臺,歡迎各類分享!