对于挑战,a fellow code golfer wrote the following code:

import java.util.*;
public class Main {
  public static void main(String[] args) {
    int size = 3;
    String[] array = new String[size];
    Arrays.fill(array, "");
    for (int i = 0; i <= 100;) {
      array[i++ % size] += i + " ";
    }
    for (String element: array) {
      System.out.println(element);
    }
  }
}

When running this code in Java 8, we get the following result:

1 4 7 10 13 16 19 22 25 28 31 34 37 40 43 46 49 52 55 58 61 64 67 70 73 76 79 82 85 88 91 94 97 100 
2 5 8 11 14 17 20 23 26 29 32 35 38 41 44 47 50 53 56 59 62 65 68 71 74 77 80 83 86 89 92 95 98 101 
3 6 9 12 15 18 21 24 27 30 33 36 39 42 45 48 51 54 57 60 63 66 69 72 75 78 81 84 87 90 93 96 99 

When running this code in Java 10, we get the following result:

2 4 6 8 10 12 14 16 18 20 22 24 26 28 30 32 34 36 38 40 42 44 46 48 50 52 54 56 58 60 62 64 66 68 70 72 74 76 78 80 82 84 86 88 90 92 94 96 98 
2 4 6 8 10 12 14 16 18 20 22 24 26 28 30 32 34 36 38 40 42 44 46 48 50 52 54 56 58 60 62 64 66 68 70 72 74 76 78 80 82 84 86 88 90 92 94 96 98 100 102 
2 4 6 8 10 12 14 16 18 20 22 24 26 28 30 32 34 36 38 40 42 44 46 48 50 52 54 56 58 60 62 64 66 68 70 72 74 76 78 80 82 84 86 88 90 92 94 96 98 100 

使用Java 10完全关闭了编号.那么这里发生了什么?这是Java10中的一个bug吗?

comments 的后续内容:

  • 当使用Java 9或更高版本编译时(我们在Java 10中发现了这个问题).在Java 8上编译这段代码,然后在Java 9或任何更高版本(包括Java 11 early access)中运行,可以得到预期的结果.

  • 这类代码是非标准的,但根据规范是有效的.它是在discussion in a golfing challenge中被Kevin Cruijssen发现的,因此遇到了奇怪的用例.

  • Didier L使用更小、更容易理解的代码简化了问题:

      class Main {
        public static void main(String[] args) {
          String[] array = { "" };
          array[test()] += "a";
        }
        static int test() {
          System.out.println("evaluated");
          return 0;
        }
      }
    

    Result when compiled in Java 8:

      evaluated
    

    Result when compiled in Java 9 and 10:

      evaluated
      evaluated
    
  • 问题似乎仅限于字符串串联和赋值运算符(+=),其左操作数为带有副作用的表达式,如array[test()]+="a"array[ix++]+="a"test()[index]+="a"test().field+="a".要启用字符串连接,至少有一个边必须具有String类型.try 在其他类型或构造上复制此操作失败.

推荐答案

这是javac中的一个bug,始于JDK 9(它对字符串连接做了一些更改,我怀疑这是问题的一部分),as confirmed by the javac team under the bug id JDK-8204322.如果查看该行对应的字节码:

array[i++%size] += i + " ";

它是:

  21: aload_2
  22: iload_3
  23: iinc          3, 1
  26: iload_1
  27: irem
  28: aload_2
  29: iload_3
  30: iinc          3, 1
  33: iload_1
  34: irem
  35: aaload
  36: iload_3
  37: invokedynamic #5,  0 // makeConcatWithConstants:(Ljava/lang/String;I)Ljava/lang/String;
  42: aastore

其中最后aaload是来自数组的实际负载.然而,这一部分

  21: aload_2             // load the array reference
  22: iload_3             // load 'i'
  23: iinc          3, 1  // increment 'i' (doesn't affect the loaded value)
  26: iload_1             // load 'size'
  27: irem                // compute the remainder

它大致对应于表达式array[i++%size](减go 实际的加载和存储),在那里有两次.这是不正确的,正如规范在jls-15.26.2中所说:

形式E1 op= E2的复合赋值表达式相当于E1 = (T) ((E1) op (E2)),其中TE1except that 103 is evaluated only once.的类型

因此,对于表达式array[i++%size] += i + " ";,部分array[i++%size]应该只计算一次.但它会被判断两次(一次用于装载,一次用于存储).

是的,这是一个错误.


一些更新:

这个错误在JDK 11中得到了修复,并被向后移植到JDK 10(herehere),但从it no longer receives public updates年起就没有被移植到JDK 9.

阿列克谢·希皮列夫(Aleksey Shipilev)在JBS page米赛道上提到了(在这里的 comments 中提到了@DidierL):

解决方法:使用-XDstringConcat=inline编译

这将恢复为使用StringBuilder进行连接,并且没有错误.

Java相关问答推荐

Java 21虚拟线程会解决转向react 式单线程框架的主要原因吗?

当我用OkHttpClient重写shouldInterceptRequest来发布数据时,Android WebView正在以纯HTML加载URL内容

空手道比赛条件

是否保证在事务性块的末尾标记违反约束?

相同的Java SerializedLambda为implMethodKind返回不同的结果

Jakarta CDI强制bean构造/注册遗留事件侦听器

计算两个浮点数之间的距离是否对称?

如何对多个字段进行分组和排序?

虚拟线程应该很快消亡吗?

try 从REST API返回对象列表时出错

try 在Android Studio中的infoWindow中使用EditText(Java)

在Frege中,我如何将一个字符串安全地转换为一个可能的Int?

根本不显示JavaFX阿拉伯字母

使用MediaPlayer类在一段时间后停止播放音乐

如何利用OpenTelemeter将初始值(零)输出到普罗米修斯

为什么我的登录终结点不能被任何请求访问?

为什么Spring要更改Java版本配置以及如何正确设置?

如何以事务方式向ibmmq发送消息

如何使用stream.allMatch()为空流返回false?

转换为JSON字符串时,日期按天递减-Java