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

对于上次【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表达式进行一个文字总结,看完之后就晓得上面的演变过程啦:

Lambda表达式作用:

  • 传递行为,而不仅仅是传值
  • 提升抽象层次
  • API重用性更好
  • 更加灵活

Java Lambda基础语法:

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

Java Lambda示例:

  • (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。

Java Lambda结构:

  • 一个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表达式的总结到这,已经对其了解得比较细了,下面还是回到这次要学习的主题上来。

Function接口:

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

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

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

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

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

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

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

然后再调用:

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

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

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

高阶函数:

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

这个在之后还会不断遇到的。

posted on 2018-01-04 16:03  cexo  阅读(1874)  评论(2编辑  收藏  举报

导航