品牌词类的网站怎么做优化,做框架模板的网站,移动互联网技术网站,机构改革网站建设PDF文档已上传Github为了支持函数式编程#xff0c;Java 8引入了Lambda表达式#xff0c;那么在Java 8中到底是如何实现Lambda表达式的呢? Lambda表达式经过编译之后#xff0c;到底会生成什么东西呢?在没有深入分析前#xff0c;让我们先想一想#xff0c;Java 8中每一…PDF文档已上传Github为了支持函数式编程Java 8引入了Lambda表达式那么在Java 8中到底是如何实现Lambda表达式的呢? Lambda表达式经过编译之后到底会生成什么东西呢?在没有深入分析前让我们先想一想Java 8中每一个Lambda表达式必须有一个函数式接口与之对应函数式接口与普通接口的区别可以参考前面的内容那么你或许在想Lambda表达式是不是转化成与之对应的函数式接口的一个实现类呢然后通过多态的方式调用子类的实现呢如下面代码是一个Lambda表达式的样例FunctionalInterfaceinterface Print{public voidprint(T x);}public classLambda {public static void PrintString(String s, Printprint) {print.print(s);}public static voidmain(String[] args) {PrintString(test, (x) -System.out.println(x));}}按照上面的分析理论上经过编译器处理后最终生成的代码应该如下面所示FunctionalInterfaceinterface Print{public voidprint(T x);}class Lambda$$0 implements Print{Overridepublic voidprint(String x) {System.out.println(x);}}public classLambda {public static voidPrintString(String s,Printprint) {print.print(s);}public static voidmain(String[] args) {PrintString(test, new Lambda$$0());}}再或者是一个内部类实现代码如下所示FunctionalInterfaceinterface Print{public voidprint(T x);}public classLambda {final class Lambda$$0 implements Print{Overridepublic voidprint(String x) {System.out.println(x);}}public static voidPrintString(String s,Printprint) {print.print(s);}public static voidmain(String[] args) {PrintString(test, new Lambda().new Lambda$$0());}}异或是这种匿名内部类实现代码如下所示:FunctionalInterfaceinterface Print{public voidprint(T x);}public classLambda {public static voidPrintString(String s,Printprint) {print.print(s);}public static voidmain(String[] args) {PrintString(test, new Print() {Overridepublic voidprint(String x) {System.out.println(x);}});}}上面的代码除了在代码长度上长了点外与用Lambda表达式实现的代码运行结果是一样的那么Java 8到底是用什么方式实现的呢?是不是上面三种实现方式中的一种呢你也许觉的自已想的是对的其实本来也就是对的在Java 8中采用的是内部类来实现Lambda表达式那么Lambda表达式到底是如何实现的呢为了探究Lambda表达式是如何实现的就得需要研究Lambda表过式最终转化成的字节码文件这就需要jdk的bin目录下的一个字节码查看工具及反编译工具javap -p Lambda.class上面命令中的-p表示输出所有类及成员运行上面的命令后得的结果如下所示:Compiled from Lambda.javapublic classLambda {publicLambda();public static void PrintString(java.lang.String, Print);public static voidmain(java.lang.String[]);private static void lambda$0(java.lang.String);}由上面的代码可以看出编译器会根据Lambda表达式生成一个私有的静态函数注意在这里说的是生成而不是等价private static void lambda$0(java.lang.String);为了验证上面的转化是否正确?我们在代码中定义一个lambda$0这个的函数最终代码如下所示FunctionalInterfaceinterface Print{public voidprint(T x);}public classLambda {public static voidPrintString(String s,Printprint) {print.print(s);}private static void lambda$0(String s) {}public static voidmain(String[] args) {PrintString(test, (x) -System.out.println(x));}}上面的代码在编译时不会报错但是运行时就会报错因为存在两个lambda$0函数如下所示是运行时的错误Exception in thread main java.lang.ClassFormatError: Duplicate method namesignature in classfile Lambdaat java.lang.ClassLoader.defineClass1(Native Method)at java.lang.ClassLoader.defineClass(ClassLoader.java:760)at java.security.SecureClassLoader.defineClass(SecureClassLoader.java:142)at java.net.URLClassLoader.defineClass(URLClassLoader.java:467)at java.net.URLClassLoader.access$100(URLClassLoader.java:73)at java.net.URLClassLoader$1.run(URLClassLoader.java:368)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.javapublic classLambda {publicLambda();public static void PrintString(java.lang.String, Print);private static void lambda$0(java.lang.String);public static voidmain(java.lang.String[]);private static void lambda$0(java.lang.String);}会发现lambda$0出现了两次那么在代码运行的时候就不知道去调用哪个因此就会抛错。有了上面的内容可以知道的是Lambda表达式在Java 8中首先会生成一个私有的静态函数这个私有的静态函数干的就是Lambda表达式里面的内容,因此上面的代码初步可以转化成如下所示的代码FunctionalInterfaceinterface Print{public voidprint(T x);}public classLambda {public static void PrintString(String s, Printprint) {print.print(s);}private static void lambda$0(String x) {System.out.println(x);}public static voidmain(String[] args) {PrintString(test, /**lambda expression**/);}}转化成上面的形式之后那么如何实现调用静态的lambda$0函数呢在这里可以在以下方法打上断点可以发现在有lambda表达式的地方运行时会进入这个函数public staticCallSite metafactory(MethodHandles.Lookup caller,String invokedName,MethodType invokedType,MethodType samMethodType,MethodHandle implMethod,MethodType instantiatedMethodType)throwsLambdaConversionException {AbstractValidatingLambdaMetafactory mf;mf newInnerClassLambdaMetafactory(caller, invokedType,invokedName, samMethodType,implMethod, instantiatedMethodType,false, EMPTY_CLASS_ARRAY, EMPTY_MT_ARRAY);mf.validateMetafactoryArgs();returnmf.buildCallSite();}在这个函数中可以发现为Lambda表达式生成了一个内部类为了验证是否生成内部类可以在运行时加上-Djdk.internal.lambda.dumpProxyClasses加上这个参数后运行时会将生成的内部类class码输出到一个文件中final class Lambda$$Lambda$1 implementsPrint {private Lambda$$Lambda$1();public voidprint(java.lang.Object);}如果运行javap -c -p则结果如下final class Lambda$$Lambda$1 implementsPrint {private Lambda$$Lambda$1();Code:0: aload_01: invokespecial #10 //Method java/lang/Object.:()V4: returnpublic voidprint(java.lang.Object);Code:0: aload_11: checkcast #14 //class java/lang/String4: invokestatic #20 //Method Lambda.lambda$0:(Ljava/lang/String;)V7: return}通过上面的字节码指令可以发现实现上调用的是Lambda.lambda$0这个私有的静态方法因此最终的Lambda表达式等价于以下形式FunctionalInterfaceinterface Print{public voidprint(T x);}public classLambda {public static void PrintString(String s, Printprint) {print.print(s);}private static void lambda$0(String x) {System.out.println(x);}final class $Lambda$1 implementsPrint{Overridepublic voidprint(Object x) {lambda$0((String)x);}}public static voidmain(String[] args) {PrintString(test, new Lambda().new $Lambda$1());}}