当前位置: 首页 > 面试题库 >

增量逻辑

许曦
2023-03-14
问题内容

我试图更深入地了解后期和前置增量器,但以下表达式有些卡住:

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
}

我知道我在某个地方缺少逻辑,但是在哪里?

我尝试过的

  • 从左到右(尽管我知道不建议这样做)
  • 从内部支架开始,然后从那里开始。

谢谢您的帮助

PS:评论是我演算的细节

编辑1

我试图将硬编码值从表达式更改为其他值,2结果总是0

看这个例子:

    int i = 0;
    i = i+=(++i + (i+=32500 + --i) - ++i);
    System.out.println(i); // Prints 0

该表达式在逻辑上应该相距不远,0但确实可以打印出来。

当我使用负数时也会发生同样的情况:

    int i = 0;
    i = i+=(++i + (i+=(-32650) + --i) - ++i);
    System.out.println(i); // Prints 0

编辑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

i无论硬编码的值是多少,它都会给出每次的两倍。


问题答案:

引用Java语言规范,15.7评估顺序:

Javahtml" target="_blank">编程语言保证运算符的操作数似乎按特定的评估顺序(即 从左到右)进行 评估。

评估 右侧操作数的任何部分之前,似乎已对二进制运算符的 左侧操作数 进行了 完全 评估。

如果该运算符是 复合赋值运算符
(第15.26.2节),则对左操作数的求值包括记住左操作数所表示的
变量 以及 获取并保存该变量的值 以用于隐式二进制操作。

因此,从本质上讲, 评估右侧 之前i += ++i会记住i左侧的旧值。 __

请记住,操作数的 求值顺序 和运算符的 优先级 是两回事。

逐步显示评估顺序,并在{大括号}中保存值:

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

跟进编辑问题

如果我们命名初始值I和常量N

int i = I;
i = i += (++i + (i += N + --i) - ++i);

然后我们可以看到这些值是:

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;


 类似资料:
  • 问题内容: 当我偶然发现名为JSpeed-Javascript优化的项目时,我正在浏览Google Code。 我注意到优化之一是更改为for循环语句。 优化之前 优化后 我知道前后的增量是什么,但是知道如何加快代码速度吗? 问题答案: 这是我阅读并可以回答你的问题:“前递增()加一的值,然后返回;相反,收益则增加了一个给它,这 _在理论上_中创建一个临时变量存储的值的结果在执行增量操作之前”。

  • 背景 : 从库Pika得到主库的全部DB结构,接下来以partition维度做Trysync,如果从库确认可以增量同步,从库将以partition为维度进行增量同步。默认使用pika port+2000的端口进行增量同步。 Binlog 结构: Pika的主从同步是使用Binlog来完成的,一主多从的结构master节点也可以给多个slave复用一个Binlog,只不过不同的slave在bingl

  • 僵尸增量是一款放置游戏,你需要控制一群僵尸破坏小镇...

  • 《复制增量》又是一款受到《反物质维度》启发的放置游戏。

  • 带领你的英雄们永无止境的冒险!目前也出了 Steam 版。

  • Incremental Transcendence  增量超越是一款网页点击/放置类游戏,你需要在其中收集资源、发展自己的家园。 点击可玩 https://static.oschina.net/trytry/Incremental-Transcendence