Java 8系列之 Lambda的函数式接口(三)

简介: Java 8系列之 Lambda的函数式接口(三)

1使用泛型


在前文实践中,我们定义了StudentFilter函数式接口,使得学生的过滤筛选函数开始支持Lambda的行为传递。其中StudentFilter定义了函数描述符为(T)->boolean类型的函数式接口,从而只要满足该描述符的行为代码都可参数化传递给学生过滤函数。


在系统开发中,布尔表达式非常常用,如果此时我们系统中需要一个(T)->boolean类型的老师筛选函数式接口,我们就需要定义一个TeacherFilter接口了吗?虽然可以这样做,但是不符合代码复用思想。如果我们能将StudentTeacher的类型也参数化,那就解决现在的问题,同时提供了统一的(T) -> boolean 函数类型的抽象接口,这里,应该很多同学应该想到了,类型参数化实质上就是泛型技术。学生过滤代码支持泛型改造后如下:

image.png



这里将Student对象进行了类型泛型化处理 ,这里读者可以与前文比对学习,本作者一直认为技术比对是一种很好的学习方式。


2

2JDK中已有的函数式接口


第一节中的OToBoolean<T>已经是一个很通用的用于布尔判断的函数式接口了,完全可以作为基础接口定义在整个项目中使用。既然这样,你就去定义吧,只不过JDK早就把你的买卖抢了。在JDK8中Java语言开发者已经将常用函数描述符的接口进行了抽象封装,定义在java.util.function 包中。例如我们定义的布尔函数式接口,在JDK中Predicate<T>接口与之具有相同的定义。这里我们详细介绍一下 java.util.function 包中的函数式接口 Predicate<T>Consumer<T>Supplier<T>Function<T,R>接口,并将在后文给出一个较全的函数式接口与描述符对应表。


Predicate<T> 接口该接口定义了一个支持泛型的boolean test( T)的抽象方法,其函数描述符为 (T)-> boolean,现在我们就可以直接使用Predicate<T>接口来替代OToBoolean接口了。


image.png



Consumer<T>接口
该接口定义了一个void accept(T)的抽象方法,其函数描述符为 (T) -> void,如果你需要一个操作某一对象,但无需返回的的函数式接口,那么就可以使用Consumer<T>接口。


image.png


Supplier<T>接口既然有消费者接口(Consumer<T>),那就要有生产者接口(Supplier<T>),该接口定义了一个 T get() 的抽象方法,其函数描述符为 () -> T,如果你需要一个从某一对象获取到某值的接口,那么就可以用Supplier<T>


image.png


Function<T,R>接口该接口定义了一个 R applay(T)类型的抽象函数,它接受一个泛型变量T,并返回一个泛型变量R,如果你需要将一个对象T映射成R,那么就可以使用Function<T,R>接口。例如我们需要打印一个花名册,但是具体打印的格式需要可灵活配置,那就将花名册打印格式参数化,需要使用 (Student) -> String 类型的函数描述符,与Function<T, R>接口吻合。


image.png

3

3关于装箱与拆箱


泛型的使用使得函数式接口有了更高的灵活性,我觉得这里应该先说一下参数化,参数化是相对于硬编码来说的 ,如我们常用的函数声明具有参数列表,lambda表达式采用了代码参数化技术,泛型则使用了类型参数化技术,参数化是代码走向通用的方法 ,同时也是编程抽象的一种体现。


为了程序员的方便,JDK中提供了现成的支持泛型的函数式接口,但是由于泛型的支持,使得接口也会存在一些性能浪费的问题。我们知道Java泛型只能支持引用类型,也就是对象,不支持原始类型(intdoublechar等),在 Java SE5之前Java程序员在泛型中使用原始类型时,只能通过其对应的引用类型(IntergerDoubleCharactor)来替换,并且需要使用函数式式的方式进行原始类型到引用类型的转换,如Integer i =newInteger(10)从 Java SE5开始,Java支持自动装箱拆箱技术,通过赋值操作,便可以将原始类型包装成引用类型如Integer i = 10,相对的自动拆箱便是将引用类型转为原始类型。


但是这样的特性也会带来牺牲性能的代价,装箱的本质是将原始类型包裹起来生成一个对象出来,并将原始类型的值保存到该对象中,相对于原始类型包装的过程和内存的占用都会相应的提高,并且在很多情况下我们使用原始类型就足够了。为此,Java 8 提供了一批避免原始类型装箱的函数式接口。例如IntPredicateIntConsumerDoublePredicateIntFunction等,使用原始类型作为接口命名的前缀便是对应的避免装箱的函数式接口声明。

查看具体声明,读者应该可以发现,这些接口的函数描述符完全没变,只是泛型使用了具体的原始类型来替代,如下:

image.png


4

4Java 8 中函数式接口列表


现在我们给出一份较为全的函数式接口与描述符对应的接口声明列表:


    函数式接口     函数描述符
Predicate<T>   (T)  -> boolean
Consumer<T>   (T)  -> void
Function< T, R >   (T)  -> R
Supplier<T>   ( )  -> T
UnaryOperator<T>    (T)  ->  T
BinaryOperator<T>   (T, T) -> T
BiPredicate<L, R>   (L, R)  -> boolean
BiConsumer<T, U>   (T, U)  -> void
BiFunction<T, U, R>   (T, U)  -> R


如果同学们需要的函数式接口没有被覆盖,可以根据JDK中的声明来编写适合自己使用的函数式接口声明。

相关文章
|
3天前
|
存储 Java API
探索Java中的Lambda表达式:现代编程的瑞士军刀
随着Java 8的推出,Lambda表达式成为了Java编程语言的一大亮点。本篇文章旨在深入探讨Lambda表达式在Java中的应用及其对现代编程实践的影响。文章首先概述Lambda表达式的基本概念和语法结构,随后通过实例分析其在函数式编程接口中的运用,最后讨论Lambda表达式如何优化代码的可读性和简洁性,以及它对Java未来发展方向的潜在影响。
|
1天前
|
Java 开发者
Java中的Lambda表达式与函数式接口
【6月更文挑战第28天】在现代的Java编程实践中,Lambda表达式和函数式接口已经成为提升代码简洁性和可读性的重要工具。本文将深入探讨Lambda表达式的基本概念、语法结构以及如何与函数式接口结合使用,旨在帮助开发者更好地理解和运用这一特性,以编写出更加优雅和高效的Java代码。
|
3天前
|
Java 编译器 开发者
Java中的Lambda表达式及其应用
在现代Java编程中,Lambda表达式作为函数式编程的重要特性之一,极大地简化了代码的书写和理解。本文将深入探讨Lambda表达式的定义、语法结构以及在实际开发中的应用场景,帮助读者更好地理解和运用这一强大的编程工具。
|
22小时前
|
Java
Java中的Lambda表达式与函数式编程
【6月更文挑战第29天】在Java 8中引入的Lambda表达式,为Java语言带来了革命性的变化。它不仅简化了代码,还使得函数式编程范式在Java中的应用成为可能。本文将深入探讨Lambda表达式的概念、语法及其在Java中的应用,同时展示如何通过Lambda表达式实现函数式编程的核心概念,如不可变性、纯函数和高阶函数等。
|
2天前
|
Java API 数据处理
Java中的lambda表达式与Stream API:高效的函数式编程
Java中的lambda表达式与Stream API:高效的函数式编程
|
3天前
|
Java 数据处理 API
使用Java Lambda表达式高效去重:一种优雅的数据处理实践
使用Java Lambda表达式高效去重:一种优雅的数据处理实践
6 0
|
3天前
|
Java Python
【学习记录 time:】Java8 Lambda 表达式中的 forEach 如何提前终止?
【学习记录 time:】Java8 Lambda 表达式中的 forEach 如何提前终止?
5 0
|
3天前
|
Java 编译器
论Java中的抽象类与接口
论Java中的抽象类与接口
JAVA802_函数式接口的概述、Supplier供给型接口、consumer消费型接口、Predicate断定型接口、Function函数型接口(下)
③. consumer 只有输入,没有返回值 ④. Predicate有一个输入参数,返回boolean ⑤. Function:有一个输入参数,有一个输出
125 0
JAVA802_函数式接口的概述、Supplier供给型接口、consumer消费型接口、Predicate断定型接口、Function函数型接口(下)