Java Lambdas:它在JVM中是如何工作的,它是OOP吗?[已关闭]

2022-09-01 08:50:06

例如,对于匿名内部类,传递(匿名)对象引用并执行该对象的方法。

Lambda 是将在需要时执行的代码块。

当遇到 lambda 时,JVM 中会发生什么情况?JVM在哪里存储与lambda相关的代码块(堆:年轻,旧或永久一代)?

我尝试搜索,我得到了使用lambdas的语法,但无法理解JVM内部发生了什么,因为在JAVA中,一切都是基于对象的。

  1. 那么在 OOP 的上下文中,lambdas 是如何工作的呢?

  2. lambdas 是否违反了 OOP 概念?

  3. Lambda 是否适合垃圾回收器,因为不会创建任何对象,因此无需担心内存问题和清除内存?


答案 1

我不会浪费时间思考 lambda 表达式是否违反了 OO 原则。它的目标是增加语言的能力,而不是编写OO代码,我不明白lambdas如何违反封装,继承或多态性。

本文介绍了 Java 如何处理 lambda 表达式:

Lambda 表达式的有趣之处在于,从 JVM 的角度来看,它们是完全不可见的。它不知道什么是匿名函数或 Lambda 表达式。它只知道字节码,这是一个严格的OO规范。这取决于语言及其编译器的创建者在这些约束下工作,以创建更新,更高级的语言元素。

请考虑以下代码:

List names = Arrays.asList("1", "2", "3");
Stream lengths = names.stream().map(name -> name.length());

...它从加载名称var开始并调用其方法非常简单,但随后它做了一些非常优雅的事情。它不是创建将包装 Lambda 函数的新对象,而是使用 Java 7 中添加的新指令将此调用站点动态链接到实际的 Lambda 函数。.stream()invokeDynamic

aload_1 //load the names var

// call its stream() func
invokeinterface java/util/List.stream:()Ljava/util/stream/Stream;

//invokeDynamic magic!
invokedynamic #0:apply:()Ljava/util/function/Function;

//call the map() func
invokeinterface java/util/stream/Stream.map:
(Ljava/util/function/Function;)Ljava/util/stream/Stream;

InvokeDynamic是在 Java 7 中添加的一条指令,用于降低 JVM 的严格性,并允许动态语言在运行时绑定符号,而不是在 JVM 编译代码时静态执行所有链接。

Lambda 代码

aload_0
invokevirtual java/lang/String.length:()
invokestatic java/lang/Integer.valueOf:(I)Ljava/lang/Integer;
areturn

答案 2

Lambda 表达式不会转换为 ,它们使用 Java 7 中引入的 invoke dynamic 来执行函数式方法。看看这个anonymous inner classes

他们违反吗?我不认为你应该在乎。Lambdas 使您的代码不那么冗长、更易于理解,并且“更容易”并行化。这就是你应该关心的。OOP

来自Brain Goetz评论:

我们不是通过编写面向对象程序或函数式程序来获得报酬,而是通过编写工作程序来获得报酬。