zoukankan      html  css  js  c++  java
  • 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是提前就定义好了行为之后,之后直接调用,这就是函数式编程跟面向对象编程的一个很大的区别,举这个例子就是用来体会这个区别的。

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

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

    高阶函数:

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

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

  • 相关阅读:
    对List进行子查询及分组
    网格数据库架构设计构想
    推荐一款简单实用的漏洞测试工具:Paros
    用友U9产品SOA设计架构遭技术质疑
    超简单使用MemCached
    不按常规出招
    如何学好AJax、求高手指点
    Begin
    SpringBoot基础
    SpringBoot基础学习(番外9.1)Spring MVC或Spring Boot配置默认访问页面不生效?
  • 原文地址:https://www.cnblogs.com/webor2006/p/8193855.html
Copyright © 2011-2022 走看看