「Java 8 函数式编程」读书笔记——lambda表达式

2017-02-05 高悦翔 更多博文 » 博客 » GitHub »

原文链接 http://blog.gaoyuexiang.cn/Java_8_Lambdas_Functional_Programming_Note_Lambda_Expression/
注:以下为加速网络访问所做的原文缓存,经过重新格式化,可能存在格式方面的问题,或偶有遗漏信息,请以原文为准。


本文是「Java 8 函数式编程」第二章的读书笔记。

Lambda引入的变化

Lambda表达式,是一种紧凑的、传递行为的方式,从编程思想上来讲,就是代码即数据

过去的Java中,存在大量的匿名内部类的使用,会新建一个匿名内部类传入调用的方法中。这种传统的方式,会造成冗余的、不易阅读的代码。

于是Lambda诞生了。Lambda的语法简化了使用匿名内部类时的模板代码,让程序员专注于编写想要执行的行为,也让代码更加简洁易读。

Lambda表达式的形式

Runnable runable = () -> System.out.println("Hello Lambda");//1
runable = () -> {
  System.out.print("Hello");
  System.out.println(" Lambda");
};//2
ActionListener listener = event -> System.out.println("get event");//3
BinaryOperator<Long> add = (x, y) -> x + y;//4
BinaryOperator<Long> minux = (Long x, Long y) -> x - y;//5

常见的Lambda表达式有以上5种,每个Lambda表达式都可以分为三个部分:

  • 参数部分:() event (x, y) (Long x, Long y)
  • 将参数和表达式主体分开的符号:->
  • 表达式主体

参数的形式

Lambda表达式可以看作是匿名内部类的简写形式,参数也就是使用匿名内部类时实现的方法的参数。

  • 有的方法不需要参数,如Runnablerun方法,所以使用()代表参数部分。
  • 有的方法只需要一个参数且类型确定,如ActionListener.actionPerformed方法,可以直接使用参数,不需要指定类型,也不需要加括号。
  • 有多个参数时,必须要加上括号,把参数扩起来
  • 当声明参数类型时,无论有多少个参数,都需要加括号

表达式主体的形式

表达式可以只有一行代码,也可以有多行代码;有的表达式有返回值,有的没有。

  • 只有一行代码的表达式不需要{}
    • 如果有返回值,不用写return,表达式会把这行代码的返回值作为返回值
    • 如果使用了{},则需要显式的写出return
  • 有多行代码的表达式必须使用{}
    • 如果有返回值,需要显式的写return

引用值,而不是变量

匿名内部类中,如果想要引用其所在方法中的变量,需要将其声明为final。这意味着你实际引用的是一个值,而不是变量。

Java 8中,虽然可以引用非final的变量,但这个变量必须是既成事实上的final,如果对变量进行修改,将无法通过编译。这意味着Lambda表达式仍然是引用的一个值,而不是变量。

实际上可以通过使用数组来绕开编译器,但是这样做之前应该考虑一下你的代码逻辑是否正确。

函数接口

只有一个抽象方法的接口叫做函数接口。

JDK中最重要的函数接口:

Interface Argument Return e.g.
Predicate T boolean fliter
Consumer T void forEach
Function T R map
Supplier None T factory function
UnaryOperator T T modify String
BinaryOperator (T, T) T add two instances

类型推断

Java 8为新成员Lambda表达式提供了类型推断的支持,在不需要声明参数类型的Lambda表达式中表现的有为明显。形如:

BinaryOperator<Integer> add = (x, y) -> x + y;

的表达式得以通过编译并正确执行,就是因为JVM通过泛型参数Integer推断出了方法参数的类型。