您的位置:首页 > 编程语言 > Java开发

java8学习之Lambda表达式继续探讨&Function接口详解

2018-01-04 16:03 567 查看
对于上次【http://www.cnblogs.com/webor2006/p/8186039.html】已经初步引入的Java8中Stream流的概念,其中使用了map的操作,它需要接受一个Function这样的函数式接口,回顾一下:



而这次专门对Function这个函数式接口进行进一步学习,因为这个函数式接口是比较重要的,先查看一下该接口的javadoc:





另外还有三个方法,两个默认方法、一个静态方法:





乍一看这些具体实现的方法貌似写得挺复杂的,各种泛形,所以前期会仔细的一个个去学习,待熟悉之后,对于之后再看到类似的实现理解起来就比较容易了,得循序渐进~

下面来实例化一个Function,还是用之前转大写的那个方式来创建:



其中有个注意的地方在上次已经说明了,这里再强调一下:



其实也很好理解,因为toUpperCase是String类中的实例方法,要想调用这个方法必定是String的实例对象去调用,所以这里可以总结一个套路:如果说是通过类这个类型后面跟着"::"之后引用的是一个类的实例方法(如:String::toUpperCase),那么,它所对应Lambda表达式的第一个参数就是调用这个方法的那个对象。当然这是未来要学习的方法引用(Method References)的创建形式之一,这里也先有个印象既可。

下面引用一个新的字符串排序的例子,将一个集合中的字符串按钮倒序排序输出,这里先用传统的方式来实现:





查看一下Comparator源码:



所以,这里改用Lambda表达式来编写:



这时看一下提示:



按照提示的做法如下:



但是由于这种建议的方式脱离了我们想要学习的Lambda表达式,所以还是将其还原成Lambda方式,置灰忽略~

其实对于目前写的这种Lambda表达式是可以简单化的,看:



如下:



还可以继续简化,对于Lambda表达式的语句体目前就一句话,可以简化为:



那上面的这些演变规则是怎么样的呢?这里继续对Lambda表达式进行一个文字总结,看完之后就晓得上面的演变过程啦:

[b]Lambda表达式作用:[/b]

传递行为,而不仅仅是传值

提升抽象层次

API重用性更好

更加灵活

[b]Java Lambda基础语法:[/b]

Java中的Lambda表达式基本语法:(argument) -> {body}
比如说:
  (arg1, arg2...) -> {body}
  (type1 arg1, type2 arg2...) -> {body}

[b]Java Lambda示例:[/b]

(int a, int b) -> {return a + b;}
接收两个整型参数,并返回一个整数。

() -> System.out.println("hello world");
不接收参数,也不返回值,打印hello world。

(String s) -> {System.out.println(s);}
接收一个字符串,并打印出该字符串,不返回值。

() -> 42
无参数,返回42。

() -> {return 3.1415};
无参数,返回3.1415。

[b]Java Lambda结构:[/b]

一个Lambda表达式可以有零个或多个参数。

参数的类型既可以明确声明,也可以根据上下文来推断。例如:(int a) 与(a)效果相同。

所有参数需要包含在圆括号内,参数之间用逗号相隔。例如:(a, b) 或 (int a, int b) 或 (String a, int b, float c)

空圆括号代表参数集为空。例如:() -> 42。

当只有一个参数,且其类型可推导时,圆括号 () 可以省略。例如:a -> return a * a。

Lambda表达式的主体可包含零条或多条语句。

如果Lambda表达式的主体只有一条语句,花括号{}可省略。匿名函数的返回类型与该主体表达式一致。例如:s -> System.out.println(s)

如果Lambda表达式的主体包含一条以上语句,则表达式必须包含在花括号{}中(形成代码块)。匿名函数的返回类型与代码块的返回类型一致,若没有返回则为空。

还有一个细节需要说明一下,看代码说话:



而如果加了return的话:



其实智能的IDE对于这种只有一条语句的完整写法也会提示我们写成expression的方式,如下:



好了,关于Lambda表达式的总结到这,已经对其了解得比较细了,下面还是回到这次要学习的主题上来。

[b]Function接口:[/b]

下面开始编写代码来用一用这个接口,首先先利用Function这个接口来定义一个方法:



这方法定议了怎么使用它呢?看着挺奇怪的,甭急,下面先来看下如何使用它,当使用之后就能体会其意义了:



那为啥结果是2呢?分析一下:







程序理解上貌似不难,但是貌似还木有看出使用Function之后的好处,下面继续再多次调用下:



下面再来写一个返回String的Function,如下:



那举这几个形式的例子要说明啥呢?有点懵,其实重点是为了说明Lambda表达式传递的并非是值,而是一种形为,那如果不用Lambda表达式的方式,改用传统的方式那代码会变成什么样子呢?



那首先得定义三个方法如下:



然后再调用:



通过一对比,有木有看出来端倪,很明显光看compute这个方法,是无法知道具体的行为的,是加、减、乘、除,都不得而知,而具体的行为是在调用的时候才传递的,而传统的方式compute2是提前就定义好了行为之后,之后直接调用,这就是函数式编程跟面向对象编程的一个很大的区别,举这个例子就是用来体会这个区别的。

另外还有一个小细节提一下,就是由于目前咱们传的行为代码都只有一行,直接传木有啥问题~~但是如果这个行为代码量非常大,那可以将这个行为单独定义好,如下:



这样的话就不至于在行为比较复杂的时候显得代码过于零乱,当然具体如何写得根据实际情况。

[b]高阶函数:[/b]

它的定义是:如果一个函数接收一个函数作为参数,或者返回一个函数作为返回值,那么该函数就叫做高阶函数。所以说像刚才我们定义的这个函数既可高阶函数:



这个在之后还会不断遇到的。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: