从内部匿名类访问外部匿名类的字段

5

如果要从内部类B访问外部类A的字段x,可以使用"A.this.x"。但如果外部类也是匿名的呢?比如:

public class Main1 {
    public static void main(String[] args) {
        Comparable c1 = new Comparable(){
            int x = 3;
            public int compareTo(Object o) {
                Comparable c2 = new Comparable(){
                    int x = 4;
                    public int compareTo(Object o) {
                        return x;  // <-- THIS LINE
                    }
                };
                return c2.compareTo(o);
            }
        };
        System.out.println(c1.compareTo(null));
    }
}

当运行这段代码时,会打印出4的值,因为c2的字段x的值为4。然而,我想更改标记为“THIS LINE”的行,以便返回外部类的x(即c1的字段x,其值为3)。如果外部类(即c1的类)是一个命名类A,那么我可以替换。
return x;

使用

return A.this.x;

但由于外部类也是匿名的,我没有可以使用的名称。

问题:是否有一种方法可以修改标记为“THIS LINE”的行,使其引用c1的字段x而不是c2的字段,而不必将匿名类更改为命名类?

我意识到这段代码非常丑陋,使用匿名类的方式也不是好的编程风格,但这段代码是由另一个程序生成的,这是实现生成器的最简单方法。

5个回答

4

我建议您选择一个名称而不是x来避免隐藏其他变量。


2
简单的答案是不要遮盖变量:
public static void main(String[] args) {
    Comparable c1 = new Comparable() {
        int x = 3;
        public int compareTo(Object o) {
            Comparable c2 = new Comparable() {
                //changed this name
                int y = 4;
                public int compareTo(Object o) {
                    return x;
                }
            };
            return c2.compareTo(o);
        }
    };
    System.out.println(c1.compareTo(null));
}

输出:

3

考虑到您正在使用生成的代码,这对您是否是一个选项?


1
据我所知,没有办法实现你想要的。如果你可以改变代码(似乎你能够),但不想将它们改成命名类,那么你能否只更改变量的名称,以便避免作用域问题?无论如何,你都应该为了清晰度而重命名它们。

1

类是匿名的 - 没有名称。无法使用名称引用访问它们的字段。没有名称。


0

我认为你必须声明这些字段为final——你现有的代码之所以编译成功,是因为你正在抱怨阴影。不是这样的。 不是真的。

这看起来像一个经典案例,使用类似Lisp的gensym将会使事情变得更容易。换句话说,让代码生成器为那些变量使用不同的标识符——例如x1和x2。


字段不需要是final的;这只适用于参数/局部变量。如果他将外部字段命名为“y”,则可以从最内部的类中访问它。 - Tim Frey

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接