本文是「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 add = (x, y) -> x + y;//4
BinaryOperator minux = (Long x, Long y) -> x - y;//5
常见的Lambda表达式有以上5种,每个Lambda表达式都可以分为三个部分:
参数部分:() event (x, y) (Long x, Long y)
将参数和表达式主体分开的符号:->
表达式主体
参数的形式
Lambda表达式可以看作是匿名内部类的简写形式,参数也就是使用匿名内部类时实现的方法的参数。
有的方法不需要参数,如Runnable的run方法,所以使用()代表参数部分。
有的方法只需要一个参数且类型确定,如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 add = (x, y) -> x + y;
的表达式得以通过编译并正确执行,就是因为JVM通过泛型参数Integer推断出了方法参数的类型。