1

我今天尝试了 Javas Hotswap,它工作得很好。在我的测试中,我偶然发现了一个相当奇怪的行为。这是我的代码:

public class Test extends JFrame implements ActionListener{

private JButton c;
private int f =1;
/**
 * @param args
 */
public static void main(String[] args) {
    Test t = new Test();
}

public Test(){
    this.setPreferredSize(new Dimension(800, 600));
    this.setDefaultCloseOperation(EXIT_ON_CLOSE);
    c = new JButton("Click");
    c.addActionListener(this);
    this.add(c);
    this.pack();
    this.setVisible(true);
}

@Override
public void actionPerformed(ActionEvent e) {
    c.setText(String.valueOf(f++));

}

}

注意c.setText(String.valueOf(f++));靠近结尾的那条线。如果我在程序运行时将其切换到f--,我注意到在更改后第一次单击按钮时,该值仍在计数。然而,之后的点击计数正确。将其改回时也会发生同样的情况。

我注意到的下一件事是:如果我将代码更改为:c.setText(String.valueOf(f+=1));,运行 JVM 并将其热交换以c.setText(String.valueOf(f-=1));直接应用更改,没有点击延迟。

我现在的问题是:究竟是什么导致了这种行为?JVM代码级别之间f++和之间有什么区别?f+=1

4

2 回答 2

0

在此代码上使用 javap:

public static void main(String[] args) throws Exception {
    int i = 0;
    i++;
    System.out.println(i);
    i+=1;
    System.out.println(i);
}

返回

  0: iconst_0
  1: istore_1
  2: iinc          1, 1
  5: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
  8: iload_1
  9: invokevirtual #3                  // Method java/io/PrintStream.println:(I)V
 12: iinc          1, 1
 15: getstatic     #2                  // Field java/lang/System.out:Ljava/io/PrintStream;
 18: iload_1
 19: invokevirtual #3                  // Method java/io/PrintStream.println:(I)V
 22: return

==> 没有区别

于 2012-04-10T15:59:09.827 回答
0

您是否关闭了所有优化?JIT 编译器可以在幕后动态执行许多不同的技巧,因此生成的底层代码执行期间多次更改

使用 actionPerformed() 您正在处理异步事件,即使没有热交换问题,也可能使执行路径变得更加困难。

做同样的测试代码

c.setText(String.valueOf(f++)); 

但是在一个带有断点集的大循环中执行它,这样你就可以用更少的移动部件来测试变化。

于 2012-04-10T15:59:42.523 回答