关于 java:为什么我的对象看到构造函数中没有给它的变量?

Why my object sees variables which were not given to it in the constructor?

我有以下代码。哪个是"正确的",我不明白:

1
2
3
4
5
6
7
8
9
private static void updateGUI(final int i, final JLabel label) {
    SwingUtilities.invokeLater(
        new Runnable() {
            public void run() {
                label.setText("You have" + i +" seconds.");
            }
        }
    );
}

我创建 Runnable 类的一个新实例,然后在该实例的 run 方法中使用变量 labeli。它有效,但我不明白它为什么有效。为什么考虑的对象会看到这些变量的值。

根据我的理解,代码应该是这样的(而且是错误的):

1
2
3
4
5
6
7
8
9
10
11
12
13
14
private static void updateGUI(final int i, final JLabel label) {
    SwingUtilities.invokeLater(new Runnable(i,label) {

        public Runnable(int i, JLabel label) {
            this.i = i;
            this.label = label;
        }

        public void run() {
            label.setText("You have" + i +" seconds.");
        }

    });
}

所以,我会将 ilabel 变量提供给构造函数,以便对象可以访问它们...

顺便说一句,在 updateGUI 中,我在 ilabel 之前使用了 final。我想我使用了 final 因为编译器想要那个。但我不明白为什么。


final 在封闭方法中声明的变量可以被该方法中的匿名内部类访问。如果您将参数设置为非最终参数,您会看到编译器抱怨。

您不能在匿名内部类中声明构造函数。人们有时通过编写初始化块来解决这个问题

1
2
3
4
5
6
7
new Runnable() {
  { /* this code is executed */ }

  public void run() {
    // ...
  }
};

我认为无法访问非最终变量的原因是因为如果您创建匿名内部类对象,然后会更改非最终变量 - 匿名内部类是否应该使用更新后的值?当函数终止执行并且变量被销毁时,它如何处理这种情况?如果它只能使用最终变量,则语义很明确:它将对最终变量的值进行快照。


因为内部类看到了外部类的所有变量。
第二个例子的问题是 Runnable 接口的实现者应该有一个没有参数的构造函数。


编译器会为您完成这项工作(因为匿名类中不能有构造函数)。这背后的基本原理是 label 位于内部类的可见区域中。您可以访问对象的字段或创建访问最终参数和局部变量的非匿名内部类。


之所以可以这样做是因为 final 关键字。这是一个解释原因的链接。 http://en.wikipedia.org/wiki/Final_(Java)