Yas*_*jaj 21 java logic pre-increment post-increment
我正试图深入了解post和pre incrementors,但是我对以下表达式有点困惑:
public static void main(String[] args) {
int i = 0;
i = i+=(++i + (i+=2 + --i) - ++i);
// i = 0 + (++i + (i+=2 + --i) - ++i);
// i = 0 + (1 + (3 + 2) - 1);
// i = 0 + (6 - 1);
System.out.println(i); // Prints 0 instead of 5
}
Run Code Online (Sandbox Code Playgroud)
我知道我在哪里错过了逻辑,但在哪里?
我尝试过的:
谢谢您的帮助
PS:评论是我的微积分的细节
编辑1
我试图将de hard编码值从表达式2
改为其他东西,结果总是给出0
看看这个例子:
int i = 0;
i = i+=(++i + (i+=32500 + --i) - ++i);
System.out.println(i); // Prints 0
Run Code Online (Sandbox Code Playgroud)
这个表达式在逻辑上应该远远不够,0
但不知何故它确实打印出来.
当我使用否定时会发生同样的情况:
int i = 0;
i = i+=(++i + (i+=(-32650) + --i) - ++i);
System.out.println(i); // Prints 0
Run Code Online (Sandbox Code Playgroud)
编辑2
现在,我将值i
改为开头:
int i = 1;
i = i+=(++i + (i+=2 + --i) - ++i);
System.out.println(i); // Prints 2
i = 2;
i = i+=(++i + (i+=10000 + --i) - ++i);
System.out.println(i); // Prints 4
i = 3;
i = i+=(++i + (i+=(-32650) + --i) - ++i);
System.out.println(i); // Prints 6
Run Code Online (Sandbox Code Playgroud)
i
无论硬编码值是什么,它都会给出每次的两倍.
And*_*eas 22
Java编程语言保证运算符的操作数似乎以特定的评估顺序进行评估,即从左到右.
在左边的操作数二元运算似乎充分评估右侧数的任何部分进行评估之前.
如果运算符是复合赋值运算符(第15.26.2节),那么对左侧操作数的计算包括记住左侧操作数表示的变量并获取并保存该变量的值以用于隐含的二进制操作.
因此,在评估右侧之前,基本上i += ++i
会记住i
左侧的旧值.
请记住,操作数的评估顺序和运算符的优先级是两回事.
逐步显示评估顺序,并在{braces}中保存值:
int i = 0;
i = i += (++i + (i += 2 + --i) - ++i); // i = 0
i{0} = i += (++i + (i += 2 + --i) - ++i); // i = 0
i{0} = i{0} += (++i + (i += 2 + --i) - ++i); // i = 0
i{0} = i{0} += (1 + (i += 2 + --i) - ++i); // i = 1
i{0} = i{0} += (1 + (i{1} += 2 + --i) - ++i); // i = 1
i{0} = i{0} += (1 + (i{1} += 2 + 0 ) - ++i); // i = 0
i{0} = i{0} += (1 + (i{1} += 2 ) - ++i); // i = 0
i{0} = i{0} += (1 + 3 - ++i); // i = 3
i{0} = i{0} += (4 - ++i); // i = 3
i{0} = i{0} += (4 - 4 ); // i = 4
i{0} = i{0} += 0 ; // i = 4
i{0} = 0 ; // i = 0
0 ; // i = 0
Run Code Online (Sandbox Code Playgroud)
跟进编辑问题
如果我们命名初始值I
和常量N
:
int i = I;
i = i += (++i + (i += N + --i) - ++i);
Run Code Online (Sandbox Code Playgroud)
然后我们可以看到值是:
i{I} = i{I} += ((I+1) + (i{I+1} += N + I) - ((I+1+N+I)+1));
i{I} = i{I} += (I + 1 + (I + 1 + N + I) - (I + 1 + N + I + 1));
i{I} = i{I} += (I + 1 + I + 1 + N + I - I - 1 - N - I - 1);
i{I} = i{I} += I;
i{I} = I + I;
i = 2 * I;
Run Code Online (Sandbox Code Playgroud)
这是考虑到您的第一次编辑(未知X
)的逻辑:
public static void main(String[] args) {
int i = 0;
i = i+=(++i + (i+=X + --i) - ++i);
// i = 0 += (++i + ((i += (X + --i)) - ++i));
// i = 0 += (1 + ((i += (X + --i)) - ++i)); // i = 1
// i = 0 += (1 + ((1 += (X + --i)) - ++i)); // i = 1 and i will then take the result of 1 += (X + --i)
// i = 0 += (1 + ((1 += (X + 0)) - ++i)); // i = 0 and i will then take the result of 1 += (X + 0)
// i = 0 += (1 + (X + 1 - ++i)); // i = X + 1
// i = 0 += (1 + (X + 1 - X - 2)); // i = X + 2
// i = 0 += (0); // i = X + 2
// i = 0;
System.out.println(i); // Prints 0
}
Run Code Online (Sandbox Code Playgroud)
诀窍在这里:
+=
是一个assignement运算符所以它是右关联的:在片段中,我添加了括号来更清楚地表达这一点++
和后缀减量操作者--
加上或减去1从值并且将结果存储回变量.+
添加剂经营者首先计算左边的操作数,然后右手操作.对于您的第二次编辑(I
添加未知):
public static void main(String[] args) {
int i = I;
i = i+=(++i + (i+=X + --i) - ++i);
// i = I += (++i + ((i += (X + --i)) - ++i));
// i = I += (I+1 + ((i += (X + --i)) - ++i)); // i = I+1
// i = I += (I+1 + ((I+1 += (X + --i)) - ++i)); // i = I+1 and i will then take the result of I+1 += (X + --i)
// i = I += (I+1 + ((I+1 += (X + I)) - ++i)); // i = I and i will then take the result of I+1 += (X + I)
// i = I += (I+1 + (X+2*I+1 - ++i)); // i = X + 2*I + 1
// i = I += (I+1 + (X+2*I+1 - X-2*I-2)); // i = X + 2*I + 2
// i = I += (I); // i = X + 2*I + 2
// i = 2 * I;
System.out.println(i); // Prints 2 * I
}
Run Code Online (Sandbox Code Playgroud)
归档时间: |
|
查看次数: |
1285 次 |
最近记录: |