|
| 1 | +# Labmda and Anonymous Classes(II) |
| 2 | + |
| 3 | +# 前言 |
| 4 | + |
| 5 | +读过上一篇之后,相信对Lambda表达式的语法以及基本原理有了一定了解。对于编写代码,有这些知识已经够用。本文将**进一步区分Lambda表达式和匿名内部类在JVM层面的区别,如果对这一部分不感兴趣,可以跳过**。 |
| 6 | + |
| 7 | +# 不仅是匿名内部类的简写 |
| 8 | + |
| 9 | +经过第一篇的的介绍,我们看到Lambda表达式似乎只是为了简化匿名内部类书写,这看起来仅仅通过语法糖在编译阶段把所有的Lambda表达式替换成匿名内部类就可以了。但实时并非如此。在JVM层面,Lambda表达式和匿名内部类有着明显的差别。 |
| 10 | + |
| 11 | +## 匿名内部类实现 |
| 12 | + |
| 13 | +**匿名内部类仍然是一个类,只是不需要程序员显示指定类名,编译器会自动为该类取名**。因此如果有如下形式的代码,编译之后将会产生两个class文件: |
| 14 | + |
| 15 | +```java |
| 16 | +public class MainAnonymousClass { |
| 17 | + public static void main(String[] args) { |
| 18 | + new Thread(new Runnable(){ |
| 19 | + @Override |
| 20 | + public void run(){ |
| 21 | + System.out.println("Anonymous Class Thread run()"); |
| 22 | + } |
| 23 | + }).start();; |
| 24 | + } |
| 25 | +} |
| 26 | +``` |
| 27 | +编译之后文件分布如下,两个class文件分别是主类和匿名内部类产生的: |
| 28 | + |
| 29 | + |
| 30 | + |
| 31 | +进一步分析主类MainAnonymousClass.class的字节码,可发现其创建了匿名内部类的对象: |
| 32 | + |
| 33 | +```java |
| 34 | +// javap -c MainAnonymousClass.class |
| 35 | +public class MainAnonymousClass { |
| 36 | + ... |
| 37 | + public static void main(java.lang.String[]); |
| 38 | + Code: |
| 39 | + 0: new #2 // class java/lang/Thread |
| 40 | + 3: dup |
| 41 | + 4: new #3 // class MainAnonymousClass$1 /*创建内部类对象*/ |
| 42 | + 7: dup |
| 43 | + 8: invokespecial #4 // Method MainAnonymousClass$1."<init>":()V |
| 44 | + 11: invokespecial #5 // Method java/lang/Thread."<init>":(Ljava/lang/Runnable;)V |
| 45 | + 14: invokevirtual #6 // Method java/lang/Thread.start:()V |
| 46 | + 17: return |
| 47 | +} |
| 48 | + |
| 49 | +``` |
| 50 | +## Lambda表达式实现 |
| 51 | + |
| 52 | +**Lambda表达式通过*invokedynamic*指令实现,书写Lambda表达式不会产生新的类**。如果有如下代码,编译之后只有一个class文件: |
| 53 | + |
| 54 | +```java |
| 55 | +public class MainLambda { |
| 56 | + public static void main(String[] args) { |
| 57 | + new Thread( |
| 58 | + () -> System.out.println("Lambda Thread run()") |
| 59 | + ).start();; |
| 60 | + } |
| 61 | +} |
| 62 | +``` |
| 63 | +编译之后的结果: |
| 64 | + |
| 65 | + |
| 66 | + |
| 67 | +通过javap反编译命名,我们更能看出Lambda表达式内部表示的不同: |
| 68 | + |
| 69 | +```java |
| 70 | +// javap -c -p MainLambda.class |
| 71 | +public class MainLambda { |
| 72 | + ... |
| 73 | + public static void main(java.lang.String[]); |
| 74 | + Code: |
| 75 | + 0: new #2 // class java/lang/Thread |
| 76 | + 3: dup |
| 77 | + 4: invokedynamic #3, 0 // InvokeDynamic #0:run:()Ljava/lang/Runnable; /*使用invokedynamic指令调用*/ |
| 78 | + 9: invokespecial #4 // Method java/lang/Thread."<init>":(Ljava/lang/Runnable;)V |
| 79 | + 12: invokevirtual #5 // Method java/lang/Thread.start:()V |
| 80 | + 15: return |
| 81 | + |
| 82 | + private static void lambda$main$0(); /*Lambda表达式变成了主类的私有方法*/ |
| 83 | + Code: |
| 84 | + 0: getstatic #6 // Field java/lang/System.out:Ljava/io/PrintStream; |
| 85 | + 3: ldc #7 // String Lambda Thread run() |
| 86 | + 5: invokevirtual #8 // Method java/io/PrintStream.println:(Ljava/lang/String;)V |
| 87 | + 8: return |
| 88 | +} |
| 89 | + |
| 90 | +``` |
| 91 | + |
| 92 | +反编译之后我们发现Lambda表达式被封装成了主类的一个私有方法,并通过*invokedynamic*指令进行调用。 |
| 93 | + |
| 94 | + |
| 95 | + |
| 96 | + |
0 commit comments