封闭类与声明类

2022-09-01 23:08:48

是否有任何情况可能会产生与 ?Class.getDeclaringClassClass.getEnclosingClass

我认为这可能与外部类的子类实例化未声明为静态的内部类有关,但我无法以这种方式获得差异:

public class Main {
  private static class StaticInnerClass {

  }

  private class MemberInnerClass {

  }

  private static class ChildClass extends Main {

  }

  public MemberInnerClass getMemberInnerClassInstance() {
    return new MemberInnerClass();
  }

  public static void main(String[] args) {
    System.out.println( StaticInnerClass.class.getDeclaringClass() );
    System.out.println( StaticInnerClass.class.getEnclosingClass() );
    System.out.println( MemberInnerClass.class.getDeclaringClass() );
    System.out.println( MemberInnerClass.class.getEnclosingClass() );
    System.out.println( new ChildClass().getMemberInnerClassInstance().getClass().getEnclosingClass() );
    System.out.println( new ChildClass().getMemberInnerClassInstance().getClass().getDeclaringClass() );
  }
}

输出:

class Main
class Main
class Main
class Main
class Main
class Main

答案 1

在这里找到 http://kickjava.com/1139.htm#ixzz1mv2nEWg7

“getDeclaringClass的微妙之处在于,在Java语言规范中,匿名内部类不算作类的成员,而命名的内部类则算作类的成员。因此,此方法为匿名类返回 null。另一种方法getEnclosingClass适用于匿名类和命名类。

例如:

public class Test {
    public static void main(String[] args) {
        new Object() {
            public void test() {
                System.out.println(this.getClass().getDeclaringClass()); //null
                System.out.println(this.getClass().getEnclosingClass()); //not null
            }
        }.test();
    }
}

这同样适用于方法作用域中的非匿名类:

class Foo {
  Class<?> bar() throws NoSuchFieldException {
    class Bar<S> { }
    return Bar.class;
  }

  static void main(String[] args) throws NoSuchFieldException {
    System.out.println(new Foo<Void>().bar().getDeclaringClass()); // null
    System.out.println(new Foo<Void>().bar().getEnclosinglass()); // Foo
  }
}

答案 2