Java8的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());
  4.     }
  5. });

而 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 names = Arrays.asList("Alice", "Bob", "Charlie", "Dave");
  2. List filteredNames = names
  3.         .filter(e -> e.length() >= 4)
  4.         .into(new ArrayList());
  5. for (String name : filteredNames) {
  6.     System.out.println(name);
  7. }

这里我们有两个新方法:

java.util.functions.Block

我们可使用一个新的迭代器方法来替换 for 循环 void forEach(Block):

 
 
 
  1. List names = Arrays.asList("Alice", "Bob", "Charlie", "Dave");
  2. names
  3.    .filter(e -> e.length() >= 4)
  4.    .forEach(e -> { System.out.println(e); });

forEach() 方法是 internal iteration 的一个实例:迭代过程在 Iterable 和 Block 内部进行,每次可访问一个元素。

***的结果就是用更少的代码来处理集合:

 
 
 
  1. List names = Arrays.asList("Alice", "Bob", "Charlie", "Dave");
  2. names
  3.    .mapped(e -> { return e.length(); })
  4.    .asIterable() // returns an Iterable of BiValue elements
  5.                  // an element's key is the person's name, its value is the string 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);
  2. private static void sayHello() {
  3.         System.out.println("hello");
  4. }

或者是一个实例的方法:

 
 
 
  1. Arrays.asList("Alice", "Bob", "Charlie", "Dave").forEach(System.out::println);

我们也可以创建工程方法并将构造器引用赋值给 java.util.functions.Factory:

 
 
 
  1. Factory biscuitFactory = Biscuit::new;
  2. Biscuit biscuit = biscuitFactory.make();

***,我们创建一个引用到随意实例的例子:

 
 
 
  1. interface Accessor {
  2.         PROPERTY access(BEAN bean);
  3. }
  4. public static void main(String[] args) {
  5.         Address address = new Address("29 Acacia Road", "Tunbridge Wells");
  6.         Accessor accessor = Address::getCity;
  7.         System.out.println(accessor.access(address));
  8. }

这里我们无需绑定方法引用到某个实例,我们直接将实例做为功能接口的参数进行传递。

默认方法

直到今天的 Java ,都不可能为一个接口添加方法而不会影响到已有的实现类。而 Java 8 允许你为接口自身指定一个默认的实现:

 
 
 
  1. interface Queue {
  2.         Message read();
  3.         void delete(Message message);
  4.         void deleteAll() default {
  5.                 Message message;
  6.                 while ((message = read()) != null) {
  7.                         delete(message);
  8.                 }
  9.         }
  10. }

子接口可以覆盖默认的方法:

 
 
 
  1. interface BatchQueue extends Queue {
  2.         void setBatchSize(int batchSize);
  3.         void deleteAll() default {
  4.                 setBatchSize(100);
  5.                 Queue.super.deleteAll();
  6.         }
  7. }

或者子接口也可以通过重新声明一个没有方法体的方法来删除默认的方法:

 
 
 
  1. interface FastQueue extends Queue {
  2.         void deleteAll();
  3. }

这个将强制所有实现了 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 语法、变量捕获、类型接口和编译等内容。

英文原文:http://datumedge.blogspot.co.uk/2012/06/java-8-Lambdas.html

本文来自:http://www.oschina.net/question/12_59047

本文标题:Java8的Lambda表达式
链接地址:http://www.shufengxianlan.com/qtweb/news32/286132.html

网站建设、网络推广公司-创新互联,是专注品牌与效果的网站制作,网络营销seo公司;服务项目有等

广告

声明:本网站发布的内容(图片、视频和文字)以用户投稿、用户转载内容为主,如果涉及侵权请尽快告知,我们将会在第一时间删除。文章观点不代表本网站立场,如需处理请联系客服。电话:028-86922220;邮箱:631063699@qq.com。内容未经允许不得转载,或转载时需注明来源: 创新互联