Java为什么要引入lambda,一句话就是为了在多核时代提高并发,因为lambda是一个闭包,和外界是独立的,多个lambda可以同时在多核上运行。
lambda的两种简单的使用方法
package defaults;
import java.util.ArrayList;
import java.util.List;
public class Clazz2 {
private void test(String a) {
//
}
public static void main(String[] args) {
List<String> list = new ArrayList<>();
list.add("one");
list.add("two");
list.add("three");
//method reference
list.forEach(System.out::println);
Clazz2 tmp = new Clazz2();
list.forEach(tmp::test);
//lambda
list.forEach((s) -> System.out.println("lambda0 " + s));
list.forEach((s) -> System.out.println("lambda1 " + s));
}
}
第一种方法是用method reference。使用这种方法生成的class文件中没有lambda$0这样的函数,但是会有Bootstrap方法;
这里使用了两种类型的method reference,一种是静态方法的,一种是实例方法的,区别只是前者会生成getstatic 字节码,后者会生成如下的一段代码来获得reference:
49 new defaults.Clazz2 [1]
52 dup
53 invokespecial defaults.Clazz2() [49]
56 astore_2 [tmp]
57 aload_1 [list]
58 aload_2 [tmp]
第二种方法是使用lambda的标准定义方法,这样会在生成的class文件中生成lambda$0,和lambda$1的函数,然后通过Bootstrap方法将调用的地方和lambda函数关联起来。
其实lambda的实现就是依靠invokedynamic命令来实现的,简单的说就是在遇到lambda的时候,生成一个invokedynamic字节码,JVM在执行到这条语句的时候,就会查找对应的Bootstrapmethod表,从中找到对应的Bootstrapmethod,其实这里叫做CallSite,然后在CallSite中找到对应的MethodHandle,通过MethodHandle来包装lambda方法,并传入想要的参数,这是就结束了;下次如果执行到同一个CallSite的时候,直接返回对应的MethodHandle就可以了,不需要再次查表。
至于为什么使用invokedynamic来实现lambda,这是一个很长的话题,感兴趣的人可以google一下。
https://docs.oracle.com/javase/8/docs/api/java/lang/FunctionalInterface.html
http://java.dzone.com/articles/dark-side-lambda-expressions
http://blog.youkuaiyun.com/zxhoo/article/details/38387141