Java 8 的 lambda 表达式 详解
Java 8 预计将在 2013 年发布,Java 8 将支持 Lambda 功能,尽管该规范还在不断的变化,但是 Java 8 的开发版已经实现了对 lambda 的支持。
关于 lambda 表达式的定义请看维基百科。
该文章将带你熟悉 lambda 语法,以及使用集合 API 中的 lambda 以及相关的语言增强,本文所有的代码都是在 JDK 8 lambda build b39 编译。
功能接口
只包含一个方法的接口被称为功能接口,Lambda 表达式用用于任何功能接口适用的地方。
java.awt.event.ActionListener
就是一个功能接口,因为它只有一个方法:void actionPerformed(ActionEvent)
. 在 Java 7 中我们会编写如下代码:
1 |
button.addActionListener( new ActionListener() { |
2 |
public void actionPerformed(ActionEvent e) { |
3 |
ui.dazzle(e.getModifiers()); |
而 Java 8 中可以简化为:
1 |
button.addActionListener(e -> { ui.dazzle(e.getModifiers()); }); |
编译器知道lambda 表达式必须符合 void actionPerformed(ActionEvent)
方法的定义。看起来 lambda 实体返回 void,实际上它可以推断出参数 e 的类型是 java.awt.event.ActionEvent
.
函数集合
Java 8 的类库包含一个新的包 java.util.functions
,这个包中有很多新的功能接口,这些接口可与集合 API 一起使用。
java.util.functions.Predicate
使用谓词 (Predicate) 来筛选集合:
1 |
List<String> names = Arrays.asList( "Alice" , "Bob" , "Charlie" , "Dave" ); |
2 |
List<String> filteredNames = names |
3 |
.filter(e -> e.length() >= 4 ) |
4 |
.into( new ArrayList<String>()); |
5 |
for (String name : filteredNames) { |
6 |
System.out.println(name); |
这里我们有两个新方法:
Iterable<T> filter(Predicate<? super T>)
用于获取元素满足某个谓词返回 true 的结果
<A extends Fillable<? super T>> A into(A)
将用返回的结果填充 ArrayList
java.util.functions.Block
我们可使用一个新的迭代器方法来替换 for 循环 void forEach(Block<? super T>)
:
1 |
List<String> names = Arrays.asList( "Alice" , "Bob" , "Charlie" , "Dave" ); |
3 |
.filter(e -> e.length() >= 4 ) |
4 |
.forEach(e -> { System.out.println(e); }); |
forEach()
方法是 internal iteration 的一个实例:迭代过程在
Iterable
和
Block
内部进行,每次可访问一个元素。
最后的结果就是用更少的代码来处理集合:
1 |
List<String> names = Arrays.asList( "Alice" , "Bob" , "Charlie" , "Dave" ); |
3 |
.mapped(e -> { return e.length(); }) |
6 |
.filter(e -> e.getValue() >= 4 ) |
7 |
.sorted((a, b) -> a.getValue() - b.getValue()) |
8 |
.forEach(e -> { System.out.println(e.getKey() + '\t' + e.getValue()); }); |
这样做的优点是:
- 元素在需要的时候才进行计算
- 如果我们取一个上千个元素的集合的前三条时,其他元素就不会被映射
- 鼓励使用方法链
- 我们无需才存储中间结果来构建新的集合
- 内部迭代过程因此大多数细节
- 例如,我们可以通过下面代码来并行 map() 操作
writing myCollection.parallel().map(e ‑> e.length())
.
方法引用
我们可通过 :: 语法来引用某个方法。方法引用被认为是跟 lambda 表达式一样的,可用于功能接口所适用的地方。
我们可以引用一个静态方法:
1 |
executorService.submit(MethodReference::sayHello); |
3 |
private static void sayHello() { |
4 |
System.out.println( "hello" ); |
或者是一个实例的方法:
1 |
Arrays.asList( "Alice" , "Bob" , "Charlie" , "Dave" ).forEach(System.out::println); |
我们也可以创建工程方法并将构造器引用赋值给
java.util.functions.Factory
:
1 |
Factory<Biscuit> biscuitFactory = Biscuit:: new ; |
2 |
Biscuit biscuit = biscuitFactory.make(); |
最后,我们创建一个引用到随意实例的例子:
1 |
interface Accessor<BEAN, PROPERTY> { |
2 |
PROPERTY access(BEAN bean); |
5 |
public static void main(String[] args) { |
6 |
Address address = new Address( "29 Acacia Road" , "Tunbridge Wells" ); |
7 |
Accessor<Address, String> accessor = Address::getCity; |
8 |
System.out.println(accessor.access(address)); |
这里我们无需绑定方法引用到某个实例,我们直接将实例做为功能接口的参数进行传递。
默认方法
直到今天的 Java ,都不可能为一个接口添加方法而不会影响到已有的实现类。而 Java 8 允许你为接口自身指定一个默认的实现:
03 |
void delete(Message message); |
04 |
void deleteAll() default { |
06 |
while ((message = read()) != null ) { |
子接口可以覆盖默认的方法:
1 |
interface BatchQueue extends Queue { |
2 |
void setBatchSize( int batchSize); |
3 |
void deleteAll() default { |
5 |
Queue. super .deleteAll(); |
或者子接口也可以通过重新声明一个没有方法体的方法来删除默认的方法:
1 |
interface FastQueue extends Queue { |
这个将强制所有实现了 FastQueue 的类必须实现 deleteAll() 方法。
HotSpot 实现
lambda 不只是可以减少很多代码的编写,其字节码和运行时的实现也比 Java 7 中的匿名类的效率更高。针对每一个 lambda 表达式,编译器都会创建一个对应的形如 lambda$1() 这样的方法。这个过程被称之为 lambda body desugaring. 当遇见一个 lambda 表达式,编译器将会发起一个 invokedynamic
调用,并从目标功能接口中获取返回值。
深入阅读
本文很多内容都基于 Brian Goetz 的文章:State of the Lambda, State of the Lambda: Libraries Edition and Translation of Lambda Expressions. 这些文字详细描述了 lambda 语法、变量捕获、类型接口和编译等内容。
评论暂时关闭