13

在一行中使用两个赋值运算符时的运算顺序是什么?

public static void main(String[] args){
    int i = 0;
    int[] a = {3, 6};
    a[i] = i = 9; // this line in particular
    System.out.println(i + " " + a[0] + " " + a[1]);
}

编辑:感谢您的帖子。我知道 = 从右边取值,但是当我编译它时,我得到:

9 9 6

我以为它会是 ArrayOutOfBounds 异常,但它在移动到 9之前分配了 'a[i]' 。它只是对数组执行此操作吗?

4

4 回答 4

13

=被解析为右关联,但评估顺序是从左到右。

所以:语句被解析为a[i] = (i = 9). 然而,表达式iin 在a[i]右手边 ( i = 9) 之前被计算,when iis still 0

它相当于:

int[] #0 = a;
int #1 = i;
int #2 = 9;
i = #2;
#0[#1] = #2;
于 2012-02-25T03:28:58.013 回答
6

根据规格:

15.26 赋值运算符 赋值运算符有 12 个;所有在语法上都是右关联的(它们从右到左分组)。因此,a=b=c 表示 a=(b=c),将 c 的值赋给 b,然后将 b 的值赋给 a。

所以,a[i] = i = 9;是一样的i = 9; a[i] = i;

编辑

事实上,事实并非如此。示例测试类:

import java.util.Arrays;

public class Mkt {
  public static void main(String[] args) {
    int[] a = new int[10];
    int i = 5;
    a[i] = i = 9;

    System.out.println(Arrays.toString(a));
  }
}

样品运行:

$ javac Mkt.java && java Mkt
[0, 0, 0, 0, 0, 9, 0, 0, 0, 0]

请参阅其他答案以获取更多信息。基本上:

  • a[i] = i = 9与 相同a[i] = (i = 9)=右结合
  • 但是,操作数评估是从左到右的,根据这个

    15.7. 评估令

    Java 编程语言保证运算符的操作数看起来是以特定的评估顺序进行评估的,即从左到右。

    建议代码不要严重依赖本规范。当每个表达式最多包含一个副作用时,代码通常会更清晰,作为其最外层的操作,并且当代码不完全依赖于表达式从左到右求值的结果时出现的异常。

我复制了第二段,这里很有启发性——编写这样令人困惑的代码几乎没有意义。

我也觉得值得一试。

于 2012-02-25T03:22:56.957 回答
2

如果我没记错的话,= 运算符是右结合的;所以我会先被分配,然后是 a[i]。

于 2012-02-25T03:13:49.420 回答
0

运算符是右结合的=(正如其他人已经说过的)。通过这个测试可以很容易地证明这一点:

int i = 2;
int j = 3;
int x = i = j;
System.out.println(x); // This prints out 3.

这适用于所有类型、对象和原语。

我听到这种说法的方式是“双重分配”,因为使用上面的示例,您将 的值分配jix

于 2012-02-25T03:21:08.073 回答