为什么在使用for-each时循环变量有效地最终化? - java

情况1:在使用for-each循环时可以工作:

    private void m10(String[] arr) {
        for (String s : arr) {
            Supplier<String> supplier = () -> {
                System.out.println(s);
                return null;
            };
            supplier.get();
        }
    }

要么

    private void m10(Object[] arr) {
        for (Object s : arr) {
            Supplier<String> supplier = () -> {
                System.out.println(s);
                return null;
            };
            supplier.get();
        }
    }

情况2:它将捕获编译时错误

    private void m11(String[] arr) {
        for (int i = 0; i < arr.length; i++) {
            Supplier<String> supplier = () -> {
                System.out.println(arr[i]);
                return null;
            };
            supplier.get();
        }
    }

在情况2中,我知道变量i并不是有效的最终变量,因为其值在循环迭代之间发生了变化。但是我不明白为什么lambda可以在情况1下工作。

参考方案

s永不更改(s = ...)。因此,编译器说:“是的,理论上我们可以将其标记为final”。那就是有效的最终的意思。即您没有将其标记为final,但可以并且仍会编译。

如果您想了解增强的for循环:

for (String s : arr)

该变量不会超出for的范围,并且不会被重新分配。即它不是:

String s = null;
for (int i = 0; i < arr.length; i++) {
    s = arr[i];
    ...
}

该变量是在循环内部创建的,因此其范围仅限于循环。它不会重复使用,而是在每次迭代时都丢弃并重新创建:

for (int i = 0; i < arr.length; i++) {
    String s = arr[i];
    ...
}

仔细看两个例子。首先,您无法编写final String s = null;,因为我们在循环s = arr[i];期间重新分配了它。但是在第二个示例中我们可以,因为变量s仅在一次迭代中已知,然后又被丢弃。所以final String s = arr[i];很好。

附带说明,这也解释了循环后为什么不能使用s的原因。它是未知的并且已经被破坏,其范围仅限于循环。

Java:找到特定字符并获取子字符串 - java

我有一个字符串4.9.14_05_29_16_21,我只需要获取4.9。数字各不相同,所以我不能简单地获得此char数组的前三个元素。我必须找到最正确的.并将其子字符串化直到那里。我来自Python,因此我将展示Python的实现方法。def foobar(some_string): location = some_string.rfind('.&…

JAVA:字节码和二进制有什么区别? - java

java字节代码(已编译的语言,也称为目标代码)与机器代码(当前计算机的本机代码)之间有什么区别?我读过一些书,他们将字节码称为二进制指令,但我不知道为什么。 参考方案 字节码是独立于平台的,在Windows中运行的编译器编译的字节码仍将在linux / unix / mac中运行。机器代码是特定于平台的,如果在Windows x86中编译,则它将仅在Win…

java:继承 - java

有哪些替代继承的方法? java大神给出的解决方案 有效的Java:偏重于继承而不是继承。 (这实际上也来自“四人帮”)。他提出的理由是,如果扩展类未明确设计为继承,则继承会引起很多不正常的副作用。例如,对super.someMethod()的任何调用都可以引导您通过未知代码的意外路径。取而代之的是,持有对本来应该扩展的类的引用,然后委托给它。这是与Eric…

Java:BigInteger,如何通过OutputStream编写它 - java

我想将BigInteger写入文件。做这个的最好方式是什么。当然,我想从输入流中读取(使用程序,而不是人工)。我必须使用ObjectOutputStream还是有更好的方法?目的是使用尽可能少的字节。谢谢马丁 参考方案 Java序列化(ObjectOutputStream / ObjectInputStream)是将对象序列化为八位字节序列的一种通用方法。但…

Java DefaultSslContextFactory密钥库动态更新 - java

我有一个使用org.restlet.engine.ssl.DefaultSslContextFactory的现有应用程序和一个在服务器启动时加载的密钥库文件。我有另一个应用程序,该应用程序创建必须添加的证书服务器运行时动态地更新到密钥库文件。为此,我在代码中创建了证书和私钥,然后将其写入到目录。该目录由bash脚本监视,该脚本检查是否有新文件,如果出现,它将…