Java 8 中需要知道的4个函数式接口-Function、Consumer、Supplier、Predicate

前言

Java 8 中提供了许多函数式接口,包括Function、Consumer、Supplier、Predicate 等等。这 4 个接口就是本篇将要分享的内容,它们都位于 java.util.function 包下。

image-20230326234016738

为什么需要知道这几个函数式接口?

因为这 4 个函数式接口是 Java 8 中新增的重要接口,同时 Java 8 的 Stream 新特性,也有用到这些接口,所以学习它们可以帮助我们更好地理解 Stream 流。

也正因为这是函数式接口,所以就可以使用 Lambda 表达式来写接口的实现逻辑。而且学习的过程中可以更好地理解函数式编程的思想。

Function 接口

说明

Function 这个单词的意思就有「函数」的意思,就数学中的 y = f(x),接收一个 x 参数,通过函数 f 运算后,返回一个结果 y。

Function 接口包含四个方法:

  • apply(T t):这是 Function 接口的主要方法,它接收一个参数并返回一个结果。同时它也是唯一的抽象的方法,剩下的都是有默认实现的(Java 8 中接口的抽象方法支持默认实现)。
  • andThen(Function after):作用是将两个 Function 组合。首先执行当前函数,再执行 andThen 函数,并将当前函数的结果作为参数传递给 andThen 函数。
  • compose(Function before):同理,将两个 Function 组合,将先执行 compose 函数,再执行当前函数,并将 compose 函数的结果作为参数传递给当前函数。
  • identity(): 返回一个执行恒等转换的函数,即返回输入参数本身。

Function 接口通常用于将一个类型的值转换为另一个类型的值。

apply 方法

// Function 接口的泛型,第一个参数是入参类型,第二个参数是出参类型
// Function 接口只有一个抽象方法,就是 apply(),下面利用 Lambda 表达式实现这个抽象方法并创建 Function 对象
Function<Integer, String> function = num -> "GTA" + num;
// 将5这个参数传递给function,得到返回结果
String result = function.apply(5);
System.out.println(result); // 打印:GTA5

andThen 和 compose 方法

// 定义两个 Function 对象进行相关转换操作
Function<String, String> upperCase = s -> s.toUpperCase();
Function<String, String> addPostfix = s -> s + "5";
// 链式调用,将 gta 这个字符串参数先传递 upperCase 这个函数进行操作,然后将得到的结果传递给 addPostfix 函数进行操作,得到返回结果
String str = upperCase.andThen(addPostfix).apply("gta");
System.out.println(str); // 打印:GTA5

identify 方法

identity 方法返回一个执行恒等转换的函数,该函数将输入参数原样返回。例如:

Function<String, String> identity = Function.identity();
String result = identity.apply("hello"); // result is "hello"

Consumer 接口

说明

Consumer 这个单词的意思就有「消费者」的意思,就把入参消费了,并不会返回结果给你。

Consumer 接口包含两个方法:

  • accept(T t):该方法接受一个参数并执行一些操作。
  • andThen(Consumer after):同理,将两个 Consumer 组合,先后进行消费。

accept 方法

Consumer 接口通常用于消费一个参数然后执行一些操作。例如:

// Consumer 接口,泛型参数是入参类型,接受一个参数,并不返回结果,相当于消费了这个参数
Consumer<String> consumer = s -> System.out.println(s);
consumer.accept("我输入什么就打印什么"); // 打印:我输入什么就打印什么

andThen 方法

组合两个 Consumer:

Consumer<String> first = s -> System.out.println(s + 5);
Consumer<String> second = s -> System.out.println(s + 6);
// 先执行 first 这个 Consumer,接着执行 second 这个 Consumer
Consumer<String> combination = first.andThen(second);
combination.accept("GTA"); // 打印:GTA5 GTA6

Supplier 接口

Supplier 接口只定义了一个 get() 方法,该方法不接受任何参数并返回一个结果。

Supplier 这个单词的意思就有「供应者」的意思,给我的感觉就是生产者,不用参数,直接生产一个东西给你。

Supplier 接口通常用于生成一个值。例如:

// Supplier 接口,泛型参数是出参类型,不接受参数,但是会提供结果,相当于生产了某个东西
Supplier<String> supplier = () -> "提供一个我随便打的字符串给调用方";
String text = supplier.get();
System.out.println(text); // 打印:提供一个我随便打的字符串给调用方

Predicate 接口

说明

Predicate 这个单词的意思就有「预言,预测,谓语,谓词」的意思,就是用来预测判断的。

Predicate 接口包含四个方法:

  • test(T t):该方法接受一个参数并返回一个布尔值
  • and(Predicate other):与另一个 Predicate 进行组合,实现逻辑与操作。
  • negate():与另一个 Predicate 进行组合,实现逻辑非操作。
  • or(Predicate other):与另一个 Predicate 进行组合,实现逻辑或操作。

test 方法

Predicate 接口通常用于测试一个条件是否成立。例如:

// Predicate 接口,泛型参数是入参类型,返回布尔值
Predicate<String> predicate = s -> s.contains("god23bin");
boolean flag = predicate.test("god23bin能给你带来收获吗?");
System.out.println("god23bin能给你带来收获吗?" + flag); // 打印:god23bin能给你带来收获吗?true

and 方法

为了便于演示,这里准备两个 Predicate:

Predicate<String> startsWithA = (str) -> str.startsWith("A"); // 如果传入的字符串是A开头,则返回 true
Predicate<String> endsWithZ = (str) -> str.endsWith("Z"); // 如果传入的字符串是Z结尾,则返回 true

使用 and 进行组合,操作:

Predicate<String> startsWithAAndEndsWithZ = startsWithA.and(endsWithZ);
System.out.println(startsWithAAndEndsWithZ.test("ABCDEFZ")); // true
System.out.println(startsWithAAndEndsWithZ.test("BCDEFGH")); // false

negate 方法

使用 negate 进行组合,操作:

Predicate<String> notStartsWithA = startsWithA.negate();
System.out.println(notStartsWithA.test("ABCDEF")); // false
System.out.println(notStartsWithA.test("BCDEFGH")); // true

or 方法

使用 or 进行组合,操作:

Predicate<String> startsWithAOrEndsWithZ = startsWithA.or(endsWithZ);
System.out.println(startsWithAOrEndsWithZ.test("ABCDEF")); // true
System.out.println(startsWithAOrEndsWithZ.test("BCDEFGH")); // false

那这些接口有什么应用呢?

**在 Stream 流中就有应用上这些函数式接口。**当然,当你有相似的需求时,你自己也可以应用上这些接口。下面说下 Stream 流中的应用。

Function 接口:例如 map 方法,map 方法就是将一个类型的值转换为另一个类型的值。

// map 方法,将 T 类型的值转换成 R 类型的值
// R 是返回的 Stream 流的元素类型,T 是原先 Stream 流的元素类型
<R> Stream<R> map(Function<? super T, ? extends R> mapper);

Consumer 接口:例如 forEach 方法

// forEach 方法,遍历 Stream 流中的元素,T 类型是 Stream 流的元素类型
void forEach(Consumer<? super T> action);

Supplier 接口:例如 generate 方法

// 生成一个无限长度的 Stream 流
public static<T> Stream<T> generate(Supplier<T> s) {
    Objects.requireNonNull(s);
    return StreamSupport.stream(
        new StreamSpliterators.InfiniteSupplyingSpliterator.OfRef<>(Long.MAX_VALUE, s), false);
}

Predicate 接口:例如 filter 方法,使用 Predicate 进行过滤操作。

// 过滤出 Stream 流中,判断结果为 true 的元素
Stream<T> filter(Predicate<? super T> predicate);

最后的最后

希望各位屏幕前的靓仔靓女们给个三连!你轻轻地点了个赞,那将在我的心里世界增添一颗明亮而耀眼的星!

咱们下期再见!

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: Java 8 函数接口是一种拥有单个抽象方法的接口,它可以被隐地转换为 lambda 表达。您可以使用 @FunctionalInterface 注解声明一个接口函数接口。这样做可以帮助编译器检测到您是否正确地在接口声明了单个抽象方法。 例如: ``` @FunctionalInterface public interface Converter<F, T> { T convert(F from); } ``` 这是一个函数接口,因为它只有一个抽象方法 `convert()`。您可以使用 lambda 表达来实现这个接口: ``` Converter<String, Integer> converter = (from) -> Integer.valueOf(from); Integer converted = converter.convert("123"); System.out.println(converted); // 123 ``` 函数接口使用起来非常方便,因为它们可以被隐地转换为 lambda 表达。这使得您可以使用更简洁的代码来实现接口,而无需显地创建一个类来实现该接口。 ### 回答2: Java 8引入了函数接口,这是一种只有一个抽象方法的接口函数接口提供了一种简洁的方来定义Lambda表达,Lambda表达可以作为函数的参数使用。 函数接口可以通过使用@FunctionalInterface注解来明确标识。这个注解是可选的,但建议使用,因为它可以确保接口只有一个抽象方法,防止不必要的错误。 Java 8提供了一些内置的函数接口,用于处理常见的函数编程场景。其一些常用的函数接口包括: 1. Predicate(断言):代表一个谓词(布尔类型函数),接受一个参数,返回一个布尔值结果。 2. Consumer(消费者):代表一个消费者(接受一个参数并执行某些操作),接受一个参数,不返回任何结果。 3. Function(函数):代表一个函数(接受一个参数并返回一个结果),接受一个参数并返回一个指定类型的结果。 4. Supplier(供应者):代表一个供应者(不接受参数但返回一个结果),不接受任何参数,返回一个指定类型的结果。 这些函数接口的引入使得Java 8具备了更强大的函数编程能力,使代码更加简洁和易读。通过结合Lambda表达函数接口,我们可以实现与函数编程语言类似的编程风格,提高代码的可读性和可维护性。 ### 回答3: Java 8 引入了函数接口的概念。函数接口是一个只有一个抽象方法的接口。在 Java 8 之前,我们需要定义一个接口,并在其添加一个抽象方法,才能够当作 Lambda 表达的参数进行传递。而在 Java 8 ,我们只需使用 @FunctionalInterface 注解来标注一个接口,就可以将其定义为函数接口了。这使得我们能够更加简洁地使用 Lambda 表达函数接口可以用来支持函数编程,也可以用作 Lambda 表达的类型。而 Java 8 为我们提供了一些内置的函数接口,如 Function、PredicateConsumerSupplier 等。这些接口都定义了一些常用的函数方法,例如 Function 接口的 apply() 方法,Predicate 接口的 test() 方法。这些方法可以直接在 Lambda 表达使用。 通过使用函数接口,我们可以更加方便地使用 Lambda 表达来完成一些常见的操作。例如,我们可以使用 Predicate 接口来过滤集合的元素,使用 Function 接口来对集合元素进行转换,使用 Consumer 接口来消费集合元素,使用 Supplier 接口来生产对象等等。 总之,Java 8 的函数接口为我们提供了更加便捷的函数编程方。我们可以利用这些接口来定义、传递和使用 Lambda 表达,从而简化代码,提高开发效率。同时,函数接口也使得代码更加易读和易维护。然而,我们需要注意函数接口的设计和使用,以避免出现意外的错误。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值