)
? ? at java.net.URLClassLoader$1.run(URLClassLoader.java:362)
? ? at java.security.AccessController.doPrivileged(Native Method)
? ? at java.net.URLClassLoader.findClass(URLClassLoader.java:361)
? ? at java.lang.ClassLoader.loadClass(ClassLoader.java:424)
? ? at sun.misc.Launcher$AppClassLoader.loadClass(Launcher.java:331)
? ? at java.lang.ClassLoader.loadClass(ClassLoader.java:357)
? ? at sun.launcher.LauncherHelper.checkAndLoadMain(LauncherHelper.java:495)
通过javap对上述错误代码进行反编译,反编译之后输出的类的成员如下所示
Compiled from "Lambda.java"
public class Lambda {
? public Lambda();
? public static void PrintString(java.lang.String, Print);
? private static void lambda$0(java.lang.String);
? public static void main(java.lang.String[]);
? private static void lambda$0(java.lang.String);
}
会发现lambda$0出现了两次,那么在代码运行的时候,就不知道去调用哪个,因此就会抛错。
有了上面的内容,可以知道的是Lambda表达式在Java 8中首先会生成一个私有的静态函数,这个私有的静态函数干的就是Lambda表达式里面的内容,因此上面的代码初步可以转化成如下所示的代码
@FunctionalInterface
interface Print {
? ? public void print(T x);
}
public class Lambda {?
? ? public static void PrintString(String s, Print print) {
? ? ? ? print.print(s);
? ? }
? ?
? ? private static void lambda$0(String x) {
? ? ? ? System.out.println(x);
? ? }
? ?
? ? public static void main(String[] args) {
? ? ? ? PrintString("test", /**lambda expression**/);
? ? }
}
转化成上面的形式之后,那么如何实现调用静态的lambda$0函数呢,在这里可以在以下方法打上断点,可以发现在有lambda表达式的地方,运行时会进入这个函数
?public static CallSite metafactory(MethodHandles.Lookup caller,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? String invokedName,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? MethodType invokedType,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? MethodType samMethodType,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? MethodHandle implMethod,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? MethodType instantiatedMethodType)
? ? ? ? ? ? throws LambdaConversionException {
? ? ? ? AbstractValidatingLambdaMetafactory mf;
? ? ? ? mf = new InnerClassLambdaMetafactory(caller, invokedType,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? invokedName, samMethodType,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? implMethod, instantiatedMethodType,
? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? ? false, EMPTY_CLASS_ARRAY, EMPTY_MT_ARRAY);
? ? ? ? mf.validateMetafactoryArgs();
? ? ? ? return mf.buildCallSite();
}
在这个函数中可以发现为Lambda表达式生成了一个内部类,为了验证是否生成内部类,可以在运行时加上-Djdk.internal.lambda.dumpProxyClasses,加上这个参数后,运行时,会将生成的内部类class码输出到一个文件中
final class Lambda$$Lambda$1 implements Print {
? private Lambda$$Lambda$1();
? public void print(java.lang.Object);
}
如果运行javap -c -p 则结果如下
final class Lambda$$Lambda$1 implements Print {
? private Lambda$$Lambda$1();
? ? Code:
? ? ? 0: aload_0
? ? ? 1: invokespecial #10? ? ? ? ? ? ? ? // Method java/lang/Object."":()V
? ? ? 4: return
? public void print(java.lang.Object);
? ? Code:
? ? ? 0: aload_1
? ? ? 1: checkcast? ? #14? ? ? ? ? ? ? ? // class java/lang/String
? ? ? 4: invokestatic? #20? ? ? ? ? ? ? ? // Method Lambda.lambda$0:(Ljava/lang/String;)V
? ? ? 7: return
}
通过上面的字节码指令可以发现实现上调用的是Lambda.lambda$0这个私有的静态方法
因此最终的Lambda表达式等价于以下形式
@FunctionalInterface
interface Print {
? ? public void print(T x);
}
public class Lambda {?
? ? public static void PrintString(String s, Print print) {
? ? ? ? print.print(s);
? ? }
? ? private static void lambda$0(String x) {
? ? ? ? System.out.println(x);
? ? }
? ? final class $Lambda$1 implements Print{
? ? ? ? @Override
? ? ? ? public void print(Object x) {
? ? ? ? ? ? lambda$0((String)x);
? ? ? ? }
? ? }
? ? public static void main(String[] args) {
? ? ? ? PrintString("test", new Lambda().new $Lambda$1());
? ? }
}