网站建设规划书的空间,罗湖网站建设多少钱,苏州三笑网络科技有限公司,福州网站营销读过上一篇之后#xff0c;相信对Lambda表达式的语法以及基本原理有了一定了解。对于编写代码#xff0c;有这些知识已经够用。本文将进一步区分Lambda表达式和匿名内部类在JVM层面的区别#xff0c;如果对这一部分不感兴趣#xff0c;可以跳过。
经过第一篇的的介绍…读过上一篇之后相信对Lambda表达式的语法以及基本原理有了一定了解。对于编写代码有这些知识已经够用。本文将进一步区分Lambda表达式和匿名内部类在JVM层面的区别如果对这一部分不感兴趣可以跳过。
经过第一篇的的介绍我们看到Lambda表达式似乎只是为了简化匿名内部类书写这看起来仅仅通过语法糖在编译阶段把所有的Lambda表达式替换成匿名内部类就可以了。但实时并非如此。在JVM层面Lambda表达式和匿名内部类有着明显的差别。
匿名内部类实现
匿名内部类仍然是一个类只是不需要程序员显示指定类名编译器会自动为该类取名。因此如果有如下形式的代码编译之后将会产生两个class文件
public class MainAnonymousClass {public static void main(String[] args) {new Thread(new Runnable(){Overridepublic void run(){System.out.println(Anonymous Class Thread run());}}).start();;}
}
编译之后文件分布如下两个class文件分别是主类和匿名内部类产生的 进一步分析主类MainAnonymousClass.class的字节码可发现其创建了匿名内部类的对象
// javap -c MainAnonymousClass.class
public class MainAnonymousClass {...public static void main(java.lang.String[]);Code:0: new #2 // class java/lang/Thread3: dup4: new #3 // class MainAnonymousClass$1 /*创建内部类对象*/7: dup8: invokespecial #4 // Method MainAnonymousClass$1.init:()V11: invokespecial #5 // Method java/lang/Thread.init:(Ljava/lang/Runnable;)V14: invokevirtual #6 // Method java/lang/Thread.start:()V17: return
}
Lambda表达式实现
Lambda表达式通过invokedynamic指令实现书写Lambda表达式不会产生新的类。如果有如下代码编译之后只有一个class文件
public class MainLambda {public static void main(String[] args) {new Thread(() - System.out.println(Lambda Thread run())).start();;}
}
编译之后的结果 通过javap反编译命名我们更能看出Lambda表达式内部表示的不同
// javap -c -p MainLambda.class
public class MainLambda {...public static void main(java.lang.String[]);Code:0: new #2 // class java/lang/Thread3: dup4: invokedynamic #3, 0 // InvokeDynamic #0:run:()Ljava/lang/Runnable; /*使用invokedynamic指令调用*/9: invokespecial #4 // Method java/lang/Thread.init:(Ljava/lang/Runnable;)V12: invokevirtual #5 // Method java/lang/Thread.start:()V15: returnprivate static void lambda$main$0(); /*Lambda表达式被封装成主类的私有方法*/Code:0: getstatic #6 // Field java/lang/System.out:Ljava/io/PrintStream;3: ldc #7 // String Lambda Thread run()5: invokevirtual #8 // Method java/io/PrintStream.println:(Ljava/lang/String;)V8: return
}
反编译之后我们发现Lambda表达式被封装成了主类的一个私有方法并通过invokedynamic指令进行调用。 推论this引用的意义
既然Lambda表达式不是内部类的简写那么Lambda内部的this引用也就跟内部类对象没什么关系了。在Lambda表达式中this的意义跟在表达式外部完全一样。因此下列代码将输出两遍Hello Hoolee而不是两个引用地址。
public class Hello {Runnable r1 () - { System.out.println(this); };Runnable r2 () - { System.out.println(toString()); };public static void main(String[] args) {new Hello().r1.run();new Hello().r2.run();}public String toString() { return Hello Hoolee; }
}