我的编程空间,编程开发者的网络收藏夹
学习永远不晚

Java8之Function函数、BiFunction函数详解

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

Java8之Function函数、BiFunction函数详解

众所周知,Java8提供了一下非常重要的函数式接口。今天我们就来讲讲其中一个函数式接口-----Function接口。
下面的代码就是Function接口的全部代码。接下来我们逐个分析一下。

@FunctionalInterfacepublic interface Function<T, R>

@FunctionalInterface 表明该接口是一个函数式接口
T 代表参数,R 代表返回值。

第一个方法------R apply(T t)

示例代码:

public class FunctionTest {    public static void main(String[] args) {        FunctionTest functionTest = new FunctionTest();        System.out.println(functionTest.compute(1, value -> 2 * value));    }    public int compute(int a, Function<Integer, Integer> function) {        int result = function.apply(a);        return result;    }}

上述代码展示的就是Function接口中apply方法的使用,其中通过 functionTest.compute方法将两个参数传过去,在compute方法中,function使用参数进行运算。

第二个方法 ------- compose

这个方法是一个默认方法。为了方便在接口中实现具体逻辑,而添加的关键字。
我们可以看到compose方法有一个输入参数为Function类型的值,并且也会返回一个Function类型的值,重点我们看最后一句就会发现,它是先调用参数的apply 方法 再调用调用者的apply 方法

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

通过上面的分析,可以得出结果----12,即对参数先执行 value * value,在执行 value * 3。

System.out.println(functionTest2.compute(2, value -> value * 3, value -> value * value));public int compute(int a, Function<Integer, Integer> function1, Function<Integer, Integer> function2) {        return function1.compose(function2).apply(a);    }

第三个方法 ------ andThen

大家是不是觉得这个方法和上面的compose方法差不多,唯一的差别就是最后一句。该方法执行的顺序就是先执行调用者的apply方法,在执行参数的apply方法。(与compose方法相反)

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

示例代码
经过上面的分析,我们很容易得出结果,值为36。

System.out.println(functionTest2.compute(2, value -> value * 3, value -> value * value));public int compute2(int a, Function<Integer, Integer> function1, Function<Integer, Integer> function2) {        return function1.andThen(function2).apply(a);    }

第四个方法 ------ identity

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

该行代码因为实现了Function中唯一的抽象方法,所以通过下面这行代码实例化出了一个 “接口” 对象,同样也会输出helloworld

Function<String, String> fun1 = Function.identity();        String str1 = fun1.apply("helloWorld");        System.out.println(str1);

看完了Function,相信大家在学习Java8的时候,都看到了还有一个叫BiFunction的类,下面我们来分析一下BiFunction

什么是BiFunction

@FunctionalInterfacepublic interface BiFunction<T, U, R>

它也是一个函数式接口,它是Function 的 另一种形式,它接收两个参数(T,U),返回一个参数(R)。那么接下来我们看看这个接口里面有哪些方法吧。

第一个方法 ------ apply

R apply(T t, U u);

示例代码:

System.out.println(functionTest2.compute4(2, 3, (value1, value2) -> value1 + value2));public int compute4(int a, int b, BiFunction<Integer, Integer, Integer> biFunction) {        return biFunction.apply(a, b);    }

上述代码,使用的就是apply方法,我们发现,其实它跟Function里面的apply其实一样,也就是多了一个参数而已。接下来,我们看第二个方法。

第二个方法 ----- andThen

源码:

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

我们发现,andThen方法里面的参数是Function类型的,那为什么不能是BiFunction类型的呢。
andThen是先执行调用者的apply,也就是说他先调用BiFunction的apply方法,再去调用参数的apply方法;其中根据Java的规范,一个方法,只能有一个返回值。所以当执行参数的apply方法的时候,只能有一个参数,所以andThen方法里面的参数必须是Function类型的。

那么大家可能会问了,为什么BiFunction里面没有compose方法。
根据Function接口里面的compose方法,我们可以分析出,它是先执行参数的apply方法,在执行调用者的apply方法。
假如BiFunction里面有compose,又因为BiFunction接受两个参数,那么通过上面的分析,可以得出,它的compose方法的参数必定是BiFunction类型的,但是下一步执行调用者的apply方法的时候,因为只有一个参数,所以是无法满足的。所以BiFunction里面没有compose方法。



源码:

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;    }}

看注释有些绕,还是看例子

//例1public class FunctionDemo {    public static void main(String[] args) {        Function<Integer,Integer> root = (v) -> {            System.out.println("root apply");            return v + 1;// 12        };        Function<Integer,Integer> before = (v) -> {            System.out.println("before apply");            return v + 1;// 先得结果11,再传给root        };         System.out.println(root.compose(before).apply(10));        // before apply        // root apply        // 12    }}
//例2public class FunctionDemo {    public static void main(String[] args) {        Function<Integer,Integer> root = (v) -> {            System.out.println("root apply");            return v + 1;// 先得结果11,再传给after        };        Function<Integer,Integer> after = (v) -> {            System.out.println("after apply");            return v + 1;// 12        };         System.out.println(root.andThen(after).apply(10));        // root apply        // after apply        // 12    }}

例1和例2虽然参数一样,结果一样,但是相对与根函数root执行的顺序一个在前,一个在后。

Function与Consumer接口都有andThen方法,两者的区别是对参数的使用方式不同。每Consumer使用的是同一个原始参数,Function的参数只会被根函数使用一次,之后的函数使用的是前一个函数的结果做为参数。

了解了Function,其他的也就一目了然了:

BiFunction<T,U,R>:代表了一个接受两个输入参数的方法,并且返回一个结果DoubleFunction<R>:代表接受一个double值参数的方法,并且返回结果DoubleToIntFunction:接受一个double类型输入,返回一个int类型结果。DoubleToLongFunction:接受一个double类型输入,返回一个long类型结果IntFunction<R>:接受一个int类型输入参数,返回一个结果 。IntToDoubleFunction:接受一个int类型输入,返回一个double类型结果 。IntToLongFunction:接受一个int类型输入,返回一个long类型结果。LongFunction<R>: 接受一个long类型输入参数,返回一个结果。LongToDoubleFunction: 接受一个long类型输入,返回一个double类型结果。LongToIntFunction:接受一个long类型输入,返回一个int类型结果。ToDoubleBiFunction<T,U>:接受两个输入参数,返回一个double类型结果ToDoubleFunction<T>:接受一个输入参数,返回一个double类型结果ToIntBiFunction<T,U>:接受两个输入参数,返回一个int类型结果。ToIntFunction<T>:接受一个输入参数,返回一个int类型结果。ToLongBiFunction<T,U>:接受两个输入参数,返回一个long类型结果。ToLongFunction<T>:接受一个输入参数,返回一个long类型结果。

来源地址:https://blog.csdn.net/qq_43842093/article/details/127996018

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

Java8之Function函数、BiFunction函数详解

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

详解Java8函数式编程之收集器的应用

这篇文章主要介绍了详解Java8函数式编程之收集器的应用,收集器是一种通用的、从流生成复杂值的结构。可以使用它从流中生成List、Set、Map等集合,需要的朋友可以参考下
2023-05-15

MySQL数据库之内置函数和自定义函数 function

目录1、内置函数1.1、字符串函数1.2、时间函数1.3、数学函数1.4、其他函数2、自定义函数2.1、创建函数2.2、查看函数2.3、调用函数2.4、删除函数2.5、注意事项3、函数流程结构案例前言:函数分为两类:系统函数和自定义函数
2022-06-15

JQuery之each函数详解

JQuery的each()函数是用来遍历一个集合(如数组或对象)的方法。它可以用于循环遍历每个元素,并对每个元素执行特定的操作。语法:```$.each(collection, callback(index, value){// 对每个元素
2023-09-23

Python入门之三角函数atan2()函数详解

描述atan2() 返回给定的 X 及 Y 坐标值的反正切值。语法以下是 atan2() 方法的语法:import math math.atan2(y, x)注意:atan2()是不能直接访问的,需要导入 math 模块,然后通过 math
2022-06-04

机器学习之损失函数(Loss Function)

损失函数(Loss Function)是机器学习和深度学习中的关键概念,它用于衡量模型的预测与实际目标之间的差异或误差。损失函数的选择对于模型的训练和性能评估至关重要,不同的任务和问题通常需要不同的损失函数。 以下是一些常见的损失函数以及它
2023-08-30

PHP函数之日期时间函数date()使用详解

date()函数是我们在php开发中常碰到并且会使用到的一个日期函数,下面我来给大家介绍date()函数的一些基本扮靓和方法,有需要了解的朋友可进入参考
2022-11-15

Python入门之三角函数sin()函数实例详解

描述 sin()返回的x弧度的正弦值。 语法 以下是sin()方法的语法:importmath math.sin(x)注意:sin()是不能直接访问的,需要导入math模块,然后通过math静态对象调用该方法。 参数 x--一个数值。 返
2022-06-04

Python入门之三角函数tan()函数实例详解

描述tan() 返回x弧度的正弦值。语法以下是 tan() 方法的语法:import math math.tan(x)注意:tan()是不能直接访问的,需要导入 math 模块,然后通过 math 静态对象调用该方法。参数x -- 一个数值
2022-06-04

编程热搜

  • Python 学习之路 - Python
    一、安装Python34Windows在Python官网(https://www.python.org/downloads/)下载安装包并安装。Python的默认安装路径是:C:\Python34配置环境变量:【右键计算机】--》【属性】-
    Python 学习之路 - Python
  • chatgpt的中文全称是什么
    chatgpt的中文全称是生成型预训练变换模型。ChatGPT是什么ChatGPT是美国人工智能研究实验室OpenAI开发的一种全新聊天机器人模型,它能够通过学习和理解人类的语言来进行对话,还能根据聊天的上下文进行互动,并协助人类完成一系列
    chatgpt的中文全称是什么
  • C/C++中extern函数使用详解
  • C/C++可变参数的使用
    可变参数的使用方法远远不止以下几种,不过在C,C++中使用可变参数时要小心,在使用printf()等函数时传入的参数个数一定不能比前面的格式化字符串中的’%’符号个数少,否则会产生访问越界,运气不好的话还会导致程序崩溃
    C/C++可变参数的使用
  • css样式文件该放在哪里
  • php中数组下标必须是连续的吗
  • Python 3 教程
    Python 3 教程 Python 的 3.0 版本,常被称为 Python 3000,或简称 Py3k。相对于 Python 的早期版本,这是一个较大的升级。为了不带入过多的累赘,Python 3.0 在设计的时候没有考虑向下兼容。 Python
    Python 3 教程
  • Python pip包管理
    一、前言    在Python中, 安装第三方模块是通过 setuptools 这个工具完成的。 Python有两个封装了 setuptools的包管理工具: easy_install  和  pip , 目前官方推荐使用 pip。    
    Python pip包管理
  • ubuntu如何重新编译内核
  • 改善Java代码之慎用java动态编译

目录