关于Java 8新引入语法特性的简要说明

Java 8在语法上的主要改进就是新增了Lambda Expression以及Method Reference。由于官方网站的介绍稍显罗嗦,而且例子也有些复杂。我这里将提供一些更为浅显、直观的例子来帮助大家理解Java 8新引入的语法特性。


Java 8中的Lambda Expression与C、C++都不太一样。Apple为LLVM Clang新开发了Blocks语法特性,使得GNU99标准C编译器在Clang编译器下就能使用Lambda Expression。而C++则在C++11标准中就引入了Lambda表达式。Clang为Lambda表达式定义了一种新类型——<return type> (^ <block identifier>)(<parameter list>)。这种定义方式非常类似于函数指针,而这也很明显地表达了Lambda表达式的函数调用签名。

而C++11则使用[<capture>](<parameter list>) -> { }来定义Lambda,Java 8与之相类似。不过C++11的返回类型直接就是auto,除非使用std::function,否则你无法直接捕获具体的lambda表达式类型。


而Java 8却使用了一种与众不同的方式。你可以自己定义一个接口,然后将该接口引用指向一个Lambda表达式。此接口当然也有限制,即必须是函数接口!什么是函数接口?即一个interface仅有一个抽象方法的接口称为是函数接口(functional interface)。然后,lambda表达式的实体定义为:(<参数列表>) -> { <lambda实现> }。其中参数列表需要与函数接口中那唯一的抽象方法的参数列表吻合,而返回类型则直接取该抽象方法的返回类型。另外,Java的Lambda表达式不能像C++的lambda以及Blocks那样取外部函数的局部变量的引用,使得其内部能直接修改外部函数的局部变量。不过对于final变量是能够获取的,这个跟创建匿名类对象一样。

虽然这种形式有点奇葩,但还好,不算太过麻烦。由于Java比C++或Objective-C来真心啰嗦很多。这次有了Method Reference之后,可以简化很多设计。下面例子也会呈现这点:

package src;

import java.util.ArrayList;

interface MyLambdaFunc {
    public void call(int p);
}

class MyClass {
    
    public static void helloStaticMehtod(int a) {
        System.out.println("Static method value is: " + a);
    }
    
    public void memberMethod(int a) {
        System.out.println("Member method is: " + a);
    }
    
    public void method1(int a) {
        System.out.println("Method 1:" + (a + 1));
    }
    
    public void method2(int a) {
        System.out.println("Method 2: " + (a + 2));
    }
    
    public void method3(int a) {
        System.out.println("Method 3: " + (a + 3));
    }
}

public class Main {

    public static void main(String[] args) {
        // TODO Auto-generated method stub
        System.out.println("Hello, world");
        
        // Type inference
        ArrayList<String> arr = new ArrayList<>();
        arr.add("hello");
        
        // Lambda Expression test
        final int a = 10;
        
        MyLambdaFunc lambda = (p) -> {
            System.out.println("The value is: " + (a + p)); 
            };

        lambda.call(100);
// Method Reference test
        MyLambdaFunc methodRef = MyClass::helloStaticMehtod;
        methodRef.call(-100);
        
        MyClass mlc = new MyClass();
        methodRef = mlc::memberMethod;
        methodRef.call(-200);
        
        MyLambdaFunc methodList[] = { mlc::method1, mlc::method2, mlc::method3 };
        methodList[0].call(100);
        methodList[1].call(100);
        methodList[2].call(100);
    }
}

 

 

posted @ 2014-04-24 20:40  zenny_chen  Views(3385)  Comments(0Edit  收藏