Java Predicate及Consumer接口函数代码实现解析
Predicate函数编程
Predicate功能判断输入的对象是否符合某个条件。官方文档解释到:Determinesiftheinputobjectmatchessomecriteria.
了解Predicate接口作用后,在学习Predicate函数编程前,先看一下Java8关于Predicate的源码:
@FunctionalInterface publicinterfacePredicate{ /** *Evaluatesthispredicateonthegivenargument. * *@paramttheinputargument *@return{@codetrue}iftheinputargumentmatchesthepredicate, *otherwise{@codefalse} */ booleantest(Tt); defaultPredicate and(Predicateother){ Objects.requireNonNull(other); return(t)->test(t)&&other.test(t); } defaultPredicate negate(){ return(t)->!test(t); } defaultPredicate or(Predicateother){ Objects.requireNonNull(other); return(t)->test(t)||other.test(t); } static Predicate isEqual(ObjecttargetRef){ return(null==targetRef) ?Objects::isNull :object->targetRef.equals(object); } }
从上面代码可以发现,Java8新增了接口的默认(default)方法和(static)静态方法。在Java8以前,接口里的方法要求全部是抽象方法。但是静态(static)方法只能通过接口名调用,不可以通过实现类的类名或者实现类的对象调用;默认(default)方法只能通过接口实现类的对象来调用。
接下来主要来使用接口方法test,可以使用匿名内部类提供test()方法的实现,也可以使用lambda表达式实现test()。
体验一下Predicate的函数式编程,使用lambda实现。其测试代码如下:
@Test publicvoidtestPredicate(){ java.util.function.PredicateboolValue=x->x>5; System.out.println(boolValue.test(1));//false System.out.println(boolValue.test(6));//true }
第1行代码:定义一个Predicate实现,入参为Integer,返回传入参数与5做比较。
第2,3行代码调用第一行,传入相关参数。
Consumer函数编程
Consumer接口的文档声明如下:
Anoperationwhichacceptsasingleinputargumentandreturnsnoresult.Unlikemostotherfunctionalinterfaces,Consumerisexpectedtooperateviaside-effects.
即接口表示一个接受单个输入参数并且没有返回值的操作。不像其它函数式接口,Consumer接口期望执行带有副作用的操作(Consumer的操作可能会更改输入参数的内部状态)。
同样,在了解Consumer函数编程前,看一下Consumer源代码,其源代码如下:
@FunctionalInterface publicinterfaceConsumer{ /** *Performsthisoperationonthegivenargument. * *@paramttheinputargument */ voidaccept(Tt); /** *Returnsacomposed{@codeConsumer}thatperforms,insequence,this *operationfollowedbythe{@codeafter}operation.Ifperformingeither *operationthrowsanexception,itisrelayedtothecallerofthe *composedoperation.Ifperformingthisoperationthrowsanexception, *the{@codeafter}operationwillnotbeperformed. * *@paramaftertheoperationtoperformafterthisoperation *@returnacomposed{@codeConsumer}thatperformsinsequencethis *operationfollowedbythe{@codeafter}operation *@throwsNullPointerExceptionif{@codeafter}isnull */ defaultConsumer andThen(Consumerafter){ Objects.requireNonNull(after); return(Tt)->{accept(t);after.accept(t);}; } }
从上面代码可以看出,Consumer使用了Java8接口新特性——接口默认(default)方法。接下来使用接口方法accept,体验一下Consumer函数编程。其测试代码如下:
@Test publicvoidtestConsumer(){ Useruser=newUser("zm"); //接受一个参数 ConsumeruserConsumer=User1->User1.setName("zmChange"); userConsumer.accept(user); System.out.println(user.getName());//zmChange }
在Java8之前的实现如下:
@Test publicvoidtest(){ Useruser=newUser("zm"); this.change(user); System.out.println(user.getName());//输出zmChange } privatevoidchange(Useruser){ user.setName("zmChange"); }
Predicate和Consumer综合应用
为了详细说明Predicate和Consumer接口,通过一个学生例子:Student类包含姓名、分数以及待付费用,每个学生可根据分数获得不同程度的费用折扣。
Student类源代码:
publicclassStudent{ StringfirstName; StringlastName; Doublegrade; DoublefeeDiscount=0.0; DoublebaseFee=2000.0; publicStudent(StringfirstName,StringlastName,Doublegrade){ this.firstName=firstName; this.lastName=lastName; this.grade=grade; } publicvoidprintFee(){ DoublenewFee=baseFee-((baseFee*feeDiscount)/100); System.out.println("Thefeeafterdiscount:"+newFee); } }
然后分别声明一个接受Student对象的Predicate接口以及Consumer接口的实现类。本例子使用Predicate接口实现类的test()方法判断输入的Student对象是否拥有费用打折的资格,然后使用Consumer接口的实现类更新输入的Student对象的折扣。
publicclassPredicateConsumerDemo{ publicstaticStudentupdateStudentFee(Studentstudent,Predicatepredicate,Consumer consumer){ if(predicate.test(student)){ consumer.accept(student); } returnstudent; } }
Predicate和Consumer接口的test()和accept()方法都接受一个泛型参数。不同的是test()方法进行某些逻辑判断并返回一个boolean值,而accept()接受并改变某个对象的内部值。updateStudentFee方法的调用如下所示:
publicclassTest{ publicstaticvoidmain(String[]args){ Studentstudent1=newStudent("Ashok","Kumar",9.5); student1=updateStudentFee(student1, //LambdaexpressionforPredicateinterface student->student.grade>8.5, //LambdaexpressionforConsumerinerface student->student.feeDiscount=30.0); student1.printFee();//Thefeeafterdiscount:1400.0 Studentstudent2=newStudent("Rajat","Verma",8.0); student2=updateStudentFee(student2, //LambdaexpressionforPredicateinterface student->student.grade>=8, //LambdaexpressionforConsumerinerface student->student.feeDiscount=20.0); student2.printFee();//Thefeeafterdiscount:1600.0 } }
通过简单对Predicate接口和Consumer接口进行应用,对函数式编程有了一个直观认识。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:czq8825#qq.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。