我正在执行以下主要方法:
int secrete = 42;
for (int i = 0; i < 5; i++) {
Consumer<String> myprinter2 =
msg -> {
System.out.println("consuming " + msg + " ," + secrete);
};
myprinter2.accept(myprinter2.toString());
}上述代码的输出是:
consuming Main$$Lambda$1/1324119927@6d311334 ,42
consuming Main$$Lambda$1/1324119927@682a0b20 ,42
consuming Main$$Lambda$1/1324119927@3d075dc0 ,42
consuming Main$$Lambda$1/1324119927@214c265e ,42
consuming Main$$Lambda$1/1324119927@448139f0 ,42如果我将secrete更改为最终结果,则输出如下:
consuming Main$$Lambda$1/2003749087@41629346 ,42
consuming Main$$Lambda$1/2003749087@41629346 ,42
consuming Main$$Lambda$1/2003749087@41629346 ,42
consuming Main$$Lambda$1/2003749087@41629346 ,42
consuming Main$$Lambda$1/2003749087@41629346 ,42secrete实际上是最终的,即使我没有声明它是最终的,那么为什么每个lambda都被认为是一个新的对象,而我没有声明它是最终的呢?
发布于 2021-12-17 18:25:41
“有效的最终”在技术上并不是必需的,如果没有它,它是可以做到的。但是语言设计人员为了避免混淆而设置了这个限制,因为如果变量不断变化,lambda会看到什么值,初始值还是最新值?其他有lambda的语言没有这个限制,规范设置了对这个用例的期望。
考虑到以下代码:
import java.util.function.Consumer;
class Main {
public static void main(String args[]) {
int i = 42;
final int j = 41;
for (int k = 0; k < 5; k++) {
Consumer<String> x = msg -> System.out.printf("x=%s, i=%d%n", msg, i);
Consumer<String> y = msg -> System.out.printf("y=%s, j=%d%n", msg, j);
Consumer<String> z = msg -> System.out.printf("z=%s%n", msg);
x.accept(x.toString());
y.accept(y.toString());
z.accept(z.toString());
}
}
}当我们使用javap -c -v Main.class检查生成的字节码时,我们看到:
11: invokedynamic #7, 0 // InvokeDynamic #0:accept:(I)Ljava/util/function/Consumer;
16: astore_3
17: invokedynamic #11, 0 // InvokeDynamic #1:accept:()Ljava/util/function/Consumer;
22: astore 4
24: invokedynamic #14, 0 // InvokeDynamic #2:accept:()Ljava/util/function/Consumer;我们可以看到兰巴达是如何翻译的。相应的static方法表明,第一个lambda是一个捕获lambda,在转换后有一个整数第1参数(#71),而其他的没有。
BootstrapMethods:
0: #63 REF_invokeStatic java/lang/invoke/LambdaMetafactory.metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;
Method arguments:
#70 (Ljava/lang/Object;)V
#71 REF_invokeStatic Main.lambda$main$0:(ILjava/lang/String;)V
#74 (Ljava/lang/String;)V
1: #63 REF_invokeStatic java/lang/invoke/LambdaMetafactory.metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;
Method arguments:
#70 (Ljava/lang/Object;)V
#75 REF_invokeStatic Main.lambda$main$1:(Ljava/lang/String;)V
#74 (Ljava/lang/String;)V
2: #63 REF_invokeStatic java/lang/invoke/LambdaMetafactory.metafactory:(Ljava/lang/invoke/MethodHandles$Lookup;Ljava/lang/String;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodType;Ljava/lang/invoke/MethodHandle;Ljava/lang/invoke/MethodType;)Ljava/lang/invoke/CallSite;
Method arguments:
#70 (Ljava/lang/Object;)V
#78 REF_invokeStatic Main.lambda$main$2:(Ljava/lang/String;)V
#74 (Ljava/lang/String;)V所以,这就是兰巴达的翻译方式。您可以在这篇文章中找到更多细节。
https://stackoverflow.com/questions/70396304
复制相似问题