Lambda表达式如何在Java字节代码中进行翻译

时间:2014-04-17 12:34:36

标签: java lambda jvm java-8 compiler-theory

我正在尝试使用java中的lambda表达式创建一个示例,我正在使用官方JDK8。我的例子运行成功。但是当我试图检查编译器如何将lambda表达式转换为字节代码时,这让我有些困惑。以下是我的例子的代码: -

public class LambdaTest {
    public Integer lambdaBinaryOpertor(BinaryOperator<Integer> binaryOperator) {
        return binaryOperator.apply(60, 72);
    }

    public static void main(String[] args) {
        LambdaTest test = new LambdaTest();
        BinaryOperator<Integer> binaryOperator = (a, b) -> a*b;
        System.out.println("Additon using Lambda BinaryOperator: "+test.lambdaBinaryOpertor(binaryOperator));
    }
}

在这个Article中,他们讨论了编译器如何将lambda表达式转换为字节代码。根据这个文档,lambda表达式转换为static方法和lambda表达式声明的位置,引用lambda static方法。以下示例位于文章中:

//Source code
class A {
    public void foo() {
        List<String> list = ...
        list.forEach( s -> { System.out.println(s); } );
    }
} 

//After compile above code "translate code " 
class A {
    public void foo() {
        List<String> list = ...
        list.forEach( [lambda for lambda$1 as Block] );
    }

    static void lambda$1(String s) {
        System.out.println(s);
    }
}

我的例子运行正常并给我们适当的结果。但是当我尝试运行javap命令来检查类的字节代码时,字节代码中没有lambdas的静态方法。

c:\>javap LambdaTest
Compiled from "LambdaTest.java"
public class LambdaTest {
public LambdaTest();
public java.lang.Integer lambdaBinaryOpertor(java.util.function.BinaryOperator <java.lang.Integer>);
public static void main(java.lang.String[]);
}

在泛型的情况下,bridge方法由编译器创建,我们也将使用javap命令检查此方法,但在lambdas的情况下,没有static方法。这篇文章发表于2012年,java 8于2014年3月推出。所以我对lambda的跨国有一些疑问:

  1. 在本文发布之后,JDK 8中是否有任何新的实现用于lambda,否则我会检查lambda方法有什么问题?
  2. 编译器如何处理lambda表达式?
  3. JVM如何调用lambda exression?

1 个答案:

答案 0 :(得分:19)

使用 javap 其他参数打印有关课程的完整信息: javap -v -p -s -c

对于您的示例,lambda的源代码将是:

private static java.lang.Integer lambda$main$0(java.lang.Integer, java.lang.Integer);
    descriptor: (Ljava/lang/Integer;Ljava/lang/Integer;)Ljava/lang/Integer;
    flags: ACC_PRIVATE, ACC_STATIC, ACC_SYNTHETIC
    Code:
      stack=2, locals=2, args_size=2
         0: aload_0       
         1: invokevirtual #17                 // Method java/lang/Integer.intValue:()I
         4: aload_1       
         5: invokevirtual #17                 // Method java/lang/Integer.intValue:()I
         8: imul          
         9: invokestatic  #2                  // Method java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
        12: areturn       
      LineNumberTable:
        line 10: 0
      LocalVariableTable:
        Start  Length  Slot  Name   Signature
            0      13     0     a   Ljava/lang/Integer;
            0      13     1     b   Ljava/lang/Integer;
}