gpt4 book ai didi

java - 哪个循环的性能更好?为什么?

转载 作者:IT老高 更新时间:2023-10-28 20:30:42 26 4
gpt4 key购买 nike

String s = "";
for(i=0;i<....){
s = some Assignment;
}

for(i=0;i<..){
String s = some Assignment;
}

我再也不需要在循环外使用“s”了。第一个选项可能更好,因为每次都不会初始化新的字符串。然而,第二个会导致变量的范围被限制在循环本身。

编辑:回应 Milhous 的回答。将字符串分配给循环内的常量是没有意义的,不是吗?不,这里的“一些分配”是指从被迭代的列表中获得的变化值。

另外,问题不是因为我担心内存管理。只是想知道哪个更好。

最佳答案

有限的范围是最好的

使用第二个选项:

for ( ... ) {
String s = ...;
}

范围不影响性能

如果您反汇编从每个编译的代码(使用 JDK 的 javap 工具),您将看到循环在两种情况下都编译为完全相同的 JVM 指令。还要注意 Brian R. Bondy's “选项#3”与选项#1 相同。使用更紧凑的范围时,不会从堆栈中添加或删除任何额外的内容,并且在两种情况下都在堆栈上使用相同的数据。

避免过早初始化

这两种情况的唯一区别是,在第一个示例中,变量 s 被不必要地初始化。这是与变量声明位置不同的问题。这增加了两条浪费的指令(加载字符串常量并将其存储在堆栈帧槽中)。一个好的静态分析工具会警告您,您永远不会读取您分配给 s 的值,并且一个好的 JIT 编译器可能会在运行时忽略它。

您可以简单地通过使用空声明(即 String s;)来解决此问题,但这被认为是不好的做法,并且具有下面讨论的另一个副作用。

通常将像 null 这样的虚假值分配给变量,只是为了消除编译器错误,即读取变量但未初始化。这个错误可以被认为是变量范围太大的提示,并且它在需要接收有效值之前被声明。空声明迫使您考虑每个代码路径;不要通过分配虚假值来忽略这个有值(value)的警告。

保留堆栈槽

如前所述,虽然 JVM 指令在这两种情况下是相同的,但存在一个微妙的副作用,这使得在 JVM 级别上最好尽可能使用最有限的范围。这在该方法的“局部变量表”中可见。考虑一下如果你有多个循环,变量声明在不必要的大范围内会发生什么:

void x(String[] strings, Integer[] integers) {
String s;
for (int i = 0; i < strings.length; ++i) {
s = strings[0];
...
}
Integer n;
for (int i = 0; i < integers.length; ++i) {
n = integers[i];
...
}
}

变量 sn 可以在它们各自的循环中声明,但由于它们不是,编译器在堆栈帧中使用两个“槽”。如果它们在循环内声明,编译器可以重用相同的槽,从而使堆栈帧更小。

真正重要的是什么

但是,这些问题大多无关紧要。一个好的 JIT 编译器会发现无法读取您浪费分配的初始值,并优化分配。在这里或那里保存一个槽不会成败您的应用程序。

重要的是让您的代码可读且易于维护,在这方面,使用有限范围显然更好。变量的作用域越小,就越容易理解它的使用方式以及对代码的任何更改都会产生什么影响。

关于java - 哪个循环的性能更好?为什么?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/110083/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com