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 方法中使用变量 label 和 i。它有效,但我不明白它为什么有效。为什么考虑的对象会看到这些变量的值。
根据我的理解,代码应该是这样的(而且是错误的):
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) { public void run() { }); |
所以,我会将 i 和 label 变量提供给构造函数,以便对象可以访问它们…
顺便说一句,在 updateGUI 中,我在 i 和 label 之前使用了 final。我想我使用了 final 因为编译器想要那个。但我不明白为什么。
- 相关问题:stackoverflow.com/questions/1950807/…
final 在封闭方法中声明的变量可以被该方法中的匿名内部类访问。如果您将参数设置为非最终参数,您会看到编译器抱怨。
您不能在匿名内部类中声明构造函数。人们有时通过编写初始化块来解决这个问题
1
2 3 4 5 6 7 |
我认为无法访问非最终变量的原因是因为如果您创建匿名内部类对象,然后会更改非最终变量 – 匿名内部类是否应该使用更新后的值?当函数终止执行并且变量被销毁时,它如何处理这种情况?如果它只能使用最终变量,则语义很明确:它将对最终变量的值进行快照。
- 实际上,基本原理是 updateGUI 方法可能会在 Runnable 对象被销毁之前终止。
- 实际上,最终变量的值作为额外的隐藏参数传递给内部类构造函数,并由内部类存储在隐藏(最终)属性中。如果您使用 javap 查看字节码,您可以看到这一点。
- @Stephen,很好。感谢您的提示。
因为内部类看到了外部类的所有变量。
第二个例子的问题是 Runnable 接口的实现者应该有一个没有参数的构造函数。
编译器会为您完成这项工作(因为匿名类中不能有构造函数)。这背后的基本原理是 label 位于内部类的可见区域中。您可以访问对象的字段或创建访问最终参数和局部变量的非匿名内部类。
之所以可以这样做是因为 final 关键字。这是一个解释原因的链接。 http://en.wikipedia.org/wiki/Final_(Java)
来源:https://www.codenong.com/2484214/