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中的声明来编写适合自己使用的函数式接口声明。

相关文章
|
2天前
|
安全 Java
在 Java 中使用实现 Runnable 接口的方式创建线程
【10月更文挑战第22天】通过以上内容的介绍,相信你已经对在 Java 中如何使用实现 Runnable 接口的方式创建线程有了更深入的了解。在实际应用中,需要根据具体的需求和场景,合理选择线程创建方式,并注意线程安全、同步、通信等相关问题,以确保程序的正确性和稳定性。
|
21小时前
|
Java
Java基础(13)抽象类、接口
本文介绍了Java面向对象编程中的抽象类和接口两个核心概念。抽象类不能被实例化,通常用于定义子类的通用方法和属性;接口则是完全抽象的类,允许声明一组方法但不实现它们。文章通过代码示例详细解析了抽象类和接口的定义及实现,并讨论了它们的区别和使用场景。
|
1天前
|
Java 测试技术 API
Java零基础-接口详解
【10月更文挑战第19天】Java零基础教学篇,手把手实践教学!
8 1
|
2天前
|
Java API 数据处理
探索Java中的Lambda表达式与Stream API
【10月更文挑战第22天】 在Java编程中,Lambda表达式和Stream API是两个强大的功能,它们极大地简化了代码的编写和提高了开发效率。本文将深入探讨这两个概念的基本用法、优势以及在实际项目中的应用案例,帮助读者更好地理解和运用这些现代Java特性。
|
6天前
|
Java 开发者
在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口
【10月更文挑战第20天】在Java多线程编程中,创建线程的方法有两种:继承Thread类和实现Runnable接口。本文揭示了这两种方式的微妙差异和潜在陷阱,帮助你更好地理解和选择适合项目需求的线程创建方式。
11 3
|
6天前
|
Java
在Java多线程编程中,实现Runnable接口通常优于继承Thread类
【10月更文挑战第20天】在Java多线程编程中,实现Runnable接口通常优于继承Thread类。原因包括:1) Java只支持单继承,实现接口不受此限制;2) Runnable接口便于代码复用和线程池管理;3) 分离任务与线程,提高灵活性。因此,实现Runnable接口是更佳选择。
18 2
|
6天前
|
Java
Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口
【10月更文挑战第20天】《JAVA多线程深度解析:线程的创建之路》介绍了Java中多线程编程的基本概念和创建线程的两种主要方式:继承Thread类和实现Runnable接口。文章详细讲解了每种方式的实现方法、优缺点及适用场景,帮助读者更好地理解和掌握多线程编程技术,为复杂任务的高效处理奠定基础。
15 2
|
6天前
|
Java 开发者
Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点
【10月更文挑战第20天】Java多线程初学者指南:介绍通过继承Thread类与实现Runnable接口两种方式创建线程的方法及其优缺点,重点解析为何实现Runnable接口更具灵活性、资源共享及易于管理的优势。
16 1
|
5天前
|
Java API
[Java]Lambda表达式
本文主要介绍了Java中的Lambda表达式,包括其优化匿名内部类的方式、使用规范、内置函数式接口及方法引用等内容。文章详细解析了Lambda的基础语法、参数列表、方法体的简化规则,以及如何利用Lambda优化代码。此外,还探讨了Lambda的作用域和引用规则,强调了对局部变量、成员变量和常量的访问限制,旨在帮助读者全面理解和掌握Lambda表达式的应用。
[Java]Lambda表达式
|
6天前
|
Java 数据处理