2012-02-25 96 views
11

在一行中使用两个赋值操作符时的操作顺序是什么?Java - 操作顺序 - 在一行中使用两个赋值操作符

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例外,但它赋予“A [1]” 之前它的移动超过9.它只是为数组做?

回答

10

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

所以:该语句被解析为a[i] = (i = 9)。然而,当i仍为0时,在右侧(i = 9)之前评估i,a[i]中的表达式。

它的像等价的:

int[] #0 = a; 
int #1 = i; 
int #2 = 9; 
i = #2; 
#0[#1] = #2; 
+0

谢谢汤姆,这是有道理的。 – HSeldon 2012-02-25 03:46:34

2

如果我没有记错,=运算符是右关联的;所以我会先分配,然后是[i]。

0

=运营商是右联合的(正如其他人已经说过的)。这可以用这个测试来证明很容易:

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

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

我听说过的这种方法是“双重任务”,因为使用上面的例子,您将j的值分配给ix

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] 

请参考the other answer以获取更多信息。基本上:

  • a[i] = i = 9相同a[i] = (i = 9),如=是右结合
  • 然而,操作数评价是左到右,作为每this

    15.7。评估顺序

    Java编程语言保证操作符的操作数看起来是按照特定的评估顺序(即从左到右)进行评估的。

    建议代码不要严格依赖于此规范。当每个表达式至多包含一个副作用时,代码通常会更清晰,作为其最外层操作,并且代码并不完全依赖于由表达式的从左到右的评估导致出现哪种异常。

我复制了第二段,这在这里非常具有启发性 - 很少写这样的混淆代码是有意义的。

我还发现this值得一试。