Java之Util下Function类学习

本文详细介绍了Java8中Consumer、Predicate、Function和Supplier四大函数接口的使用方法与应用场景,包括接口定义、源码解析及典型实例。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

JAVA8的java.util.function包

Java8之Consumer、Supplier、Predicate和Function攻略

一.概述

nametypedescription
ConsumerConsumer<T>接收T对象,不返回值
PredicatePredicate<T>接收T对象并返回 boolean
FunctionFunction<T,R>接收T对象,返回R对象
SupplierSupplier<T>提供对象,不接收值

二.Consumer接口

Consumer 接口就是一个消费型的接口,通过传入参数,然后输出值。

2.1 源码

@FunctionalInterface
public interface Consumer<T> {

    void accept(T t);

    default Consumer<T> andThen(Consumer<? super T> after) {
        Objects.requireNonNull(after);
        return (T t) -> { accept(t); after.accept(t); };
    }
}

例如 Stream 中的 foreach 方法要求传的参数就是 Consumer

483

void forEach(Consumer<? super T> action);

2.2 Consumer实例

以下是网上大佬举的例子

/**
     * consumer接口测试
     */
@Test
public void test_Consumer() {
    //① 使用consumer接口实现方法
    Consumer<String> consumer = new Consumer<String>() {

        @Override
        public void accept(String s) {
            System.out.println(s);
        }
    };
    Stream<String> stream = Stream.of("aaa", "bbb", "ddd", "ccc", "fff");
    stream.forEach(consumer);

    System.out.println("********************");

    //② 使用lambda表达式,forEach方法需要的就是一个Consumer接口
    stream = Stream.of("aaa", "bbb", "ddd", "ccc", "fff");
    Consumer<String> consumer1 = (s) -> System.out.println(s);//lambda表达式返回的就是一个Consumer接口
    stream.forEach(consumer1);
    //更直接的方式
    //stream.forEach((s) -> System.out.println(s));
    System.out.println("********************");

    //③ 使用方法引用,方法引用也是一个consumer
    stream = Stream.of("aaa", "bbb", "ddd", "ccc", "fff");
    Consumer consumer2 = System.out::println;
    stream.forEach(consumer);
    //更直接的方式
    //stream.forEach(System.out::println);
}

2.3 实例分析

consumer接口分析

在代码①中,我们直接创建 Consumer 接口,并且实现了一个名为 accept 的方法,这个方法就是这个接口的关键了。

我们看一下 accept 方法;这个方法传入一个参数,不返回值。当我们发现 forEach 需要一个 Consumer 类型的参数的时候,传入之后,就可以输出对应的值了。

② lambda 表达式作为 consumer

Consumer<String> consumer1 = (s) -> System.out.println(s);//lambda表达式返回的就是一个Consumer接口

在上面的代码中,我们使用下面的 lambda 表达式作为 Consumer。仔细的看一下你会发现,lambda 表达式返回值就是一个 Consumer;所以,你也就能够理解为什么 forEach 方法可以使用 lamdda 表达式作为参数了吧。

③ 方法引用作为 consumer

Consumer consumer2 = System.out::println;

在上面的代码中,我们用了一个方法引用的方式作为一个 Consumer ,同时也可以传给 forEach 方法。

三.Predicate接口

Predicate 接口就是一个判断型接口,这个就是一个类似于 bool 类型的判断的接口。

3.1 源码

@FunctionalInterface
public interface Predicate<T> {

    boolean test(T t);

    //以下用的比较少,待研究...
    default Predicate<T> and(Predicate<? super T> other) {
        Objects.requireNonNull(other);
        return (t) -> test(t) && other.test(t);
    }

    default Predicate<T> negate() {
        return (t) -> !test(t);
    }

    default Predicate<T> or(Predicate<? super T> other) {
        Objects.requireNonNull(other);
        return (t) -> test(t) || other.test(t);
    }

    static <T> Predicate<T> isEqual(Object targetRef) {
        return (null == targetRef)
                ? Objects::isNull
                : object -> targetRef.equals(object);
    }
}

例如:Stream 中的 filter 方法

484

Stream<T> filter(Predicate<? super T> predicate);

3.2 Predicate 实例

以下实例来自网络上大佬的实例

/**
     * Predicate谓词测试,Predicate作为接口使用
     */
@Test
public void test_Predicate2() {
    //① 将Predicate作为filter接口,Predicate起到一个判断的作用
    Predicate<Integer> predicate = new Predicate<Integer>() {
        @Override
        public boolean test(Integer integer) {
            if(integer > 5){
                return true;
            }
            return false;
        }
    };

    Stream<Integer> stream = Stream.of(1, 23, 3, 4, 5, 56, 6, 6);
    List<Integer> list = stream.filter(predicate).collect(Collectors.toList());
    list.forEach(System.out::println);

    System.out.println("********************");

}

3.3 总结

① Predicate 是一个谓词型接口,其实只是起到一个判断作用。

② Predicate 通过实现一个 test 方法做判断。

四.Function 接口

Fuction 接口就是一个功能型接口,它的作用就是转换作用,将输入数据转换成另一种形式的输出数据。

4.1 源码

@FunctionalInterface
public interface Function<T, R> {

    R apply(T t);
    
    default <V> Function<V, R> compose(Function<? super V, ? extends T> before) {
        Objects.requireNonNull(before);
        return (V v) -> apply(before.apply(v));
    }

    default <V> Function<T, V> andThen(Function<? super R, ? extends V> after) {
        Objects.requireNonNull(after);
        return (T t) -> after.apply(apply(t));
    }

    static <T> Function<T, T> identity() {
        return t -> t;
    }
}

例如:Stream 中的 map 方法

485

<R> Stream<R> map(Function<? super T, ? extends R> mapper);

4.2 Function 接口实例

以下实例来自网上大佬

/**
     * Function测试,function的作用是转换,将一个值转为另外一个值
     */
@Test
public void test_Function() {
    //① 使用map方法,泛型的第一个参数是转换前的类型,第二个是转化后的类型
    Function<String, Integer> function = new Function<String, Integer>() {
        @Override
        public Integer apply(String s) {
            return s.length();//获取每个字符串的长度,并且返回
        }
    };

    Stream<String> stream = Stream.of("aaa", "bbbbb", "ccccccv");
    Stream<Integer> stream1 = stream.map(function);
    stream1.forEach(System.out::println);

    System.out.println("********************");

}

以上实例可以方便的理解 Function 的作用。

4.3 总结

① Function 接口是一个功能型接口,是一个转换数据的作用。

② Function 接口实现 apply 方法来做转换。

五.Supplier

Supplier 接口就是一个供给型 的接口,就是一个容器。

5.1 源码

@FunctionalInterface
public interface Supplier<T> {

    /**
     * Gets a result.
     *
     * @return a result
     */
    T get();
}

Optional 中有一个方法 orElseThrow 就是其应用

486

public <X extends Throwable> T orElseThrow(Supplier<? extends X> exceptionSupplier) throws X {
    if (value != null) {
        return value;
    } else {
        throw exceptionSupplier.get();
    }
}

5.2 Supplier 实例

以下来自网上的大佬的实例

**
    * Supplier接口测试,supplier相当一个容器或者变量,可以存储值
    */
    @Test
    public void test_Supplier() {
    //① 使用Supplier接口实现方法,只有一个get方法,无参数,返回一个值
    Supplier<Integer> supplier = new Supplier<Integer>() {
        @Override
        public Integer get() {
            //返回一个随机值
            return new Random().nextInt();
        }
    };

    System.out.println(supplier.get());

    System.out.println("********************");

    //② 使用lambda表达式,
    supplier = () -> new Random().nextInt();
    System.out.println(supplier.get());
    System.out.println("********************");

    //③ 使用方法引用
    Supplier<Double> supplier2 = Math::random;
    System.out.println(supplier2.get());
}

5.3 实例分析

① Supplier接口分析

Supplier<Integer> supplier = new Supplier<Integer>() {
    @Override
    public Integer get() {
        //返回一个随机值
        return new Random().nextInt();
    }
};

看一下这段代码,我们通过创建一个 Supplier 对象,实现了一个 get 方法,这个方法无参数,返回一个值;所以,每次使用这个接口的时候都会返回一个值,并且保存在这个接口中,所以说是一个容器

② lambda表达式作为 Supplier

//② 使用lambda表达式,
supplier = () -> new Random().nextInt();
System.out.println(supplier.get());
System.out.println("********************");

上面的这段代码,我们使用 lambda 表达式返回一个 Supplier类型的接口,然后,我们调用 get 方法就可以获取这个值了。

③ 方法引用作为 Supplier

//③ 使用方法引用
Supplier<Double> supplier2 = Math::random;
System.out.println(supplier2.get());

方法引用也是返回一个Supplier类型的接口。

### Java `util.function` 包功能与使用教程 #### 1. 背景介绍 自 JDK 1.8 发布以来,Java 引入了 Lambda 表达式以及支持函数式编程的相关特性。为了更好地支持这些新特性和简化开发流程,Java 提供了一个新的包——`java.util.function`[^1]。 该包主要包含了各种用于函数式编程的基础接口,它们被设计成可以作为 lambda 表达式的靶向型(target type),从而使得开发者能够更方便地编写简洁而高效的代码[^2]。 --- #### 2. 常见接口及其用途 以下是 `java.util.function` 中一些常用的函数式接口: ##### (1) **Function<T, R>** 表示接受一个参数并返回结果的函数。此接口的主要方法是 `apply(T t)`,它会应用给定的操作于输入对象,并返回计算后的结果[^3]。 ```java import java.util.function.Function; public class Example { public static void main(String[] args) { Function<String, Integer> stringLength = s -> s.length(); System.out.println(stringLength.apply("Hello")); // 输出: 5 } } ``` ##### (2) **Predicate<T>** 代表一种布尔值判断逻辑,通常用来测试某个条件是否满足。核心方法为 `test(T t)`,如果指定的对象符合内部定义的标准,则返回 true;否则 false。 ```java import java.util.function.Predicate; public class PredicateExample { public static void main(String[] args) { Predicate<Integer> isEven = n -> n % 2 == 0; System.out.println(isEven.test(4)); // 输出: true } } ``` ##### (3) **Consumer<T>** 执行只读操作而不改变数据本身的行为模式。典型的方法有 `accept(T t)` 和默认实现的 `andThen()` 方法来组合多个消费者实例一起工作。 ```java import java.util.function.Consumer; public class ConsumerExample { public static void main(String[] args) { Consumer<String> printer = str -> System.out.print(str); printer.accept("World "); // 输出: World } } ``` ##### (4) **Supplier<T>** 提供无参调用即可获取某种型的实例化对象的方式。它的唯一抽象方法叫做 `get()`,没有任何形参列表却总是要产生一个新的 T 型的结果。 ```java import java.util.Random; import java.util.function.Supplier; public class SupplierExample { public static void main(String[] args) { Supplier<Double> randomValueGenerator = () -> Math.random() * 100d; double value = randomValueGenerator.get(); System.out.printf("%.2f%n",value); // 随机数输出示例 } } ``` ##### (5) **BiFunction<T,U,R>** 似于 Function 接口但是接收两个不同型的参数而不是单一的一个。其 apply 方法签名变为 `(T t, U u)` 形式。 ```java import java.util.function.BiFunction; public class BiFunctionExample { public static void main(String[] args){ BiFunction<Integer,Integer,Integer> adder=(a,b)-> a+b ; int sum=adder.apply(7 ,9 ); System.out.println(sum );// 输出 :16 } } ``` --- #### 3. 如何解决找不到 `java.util.function.Function` 的问题? 当遇到 “无法访问 `java.util.function.Function` 找不到 `java.util.function.Function` 的文件”的错误提示时,可能是因为当前项目的编译器设置不兼容 JDK 1.8 或更高版本所致。可以通过修改 Maven 构建工具中的配置文件(pom.xml) 来调整源码级别(source level) 及目标字节码等级(target bytecode version)[^4]: ```xml <build> <plugins> <plugin> <groupId>org.apache.maven.plugins</groupId> <artifactId>maven-compiler-plugin</artifactId> <version>3.8.1</version> <configuration> <source>1.8</source> <target>1.8</target> </configuration> </plugin> </plugins> </build> ``` --- #### 4. 总结 通过学习上述内容可知,`java.util.function` 是一个非常重要的库集合,极大地增强了 Java 对现代程序设计理念的支持程度。利用其中预置的各种功能性接口配合 lambdas 技术可以让我们的编码过程更加流畅自然. ---
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值