Java内部类

阅读:771
作者:majingjing
发布:2021-04-25 18:25:42

Java 类中不仅可以定义变量和方法,还可以定义类,这样定义在类内部的类就被称为内部类。根 据定义的方式不同,内部类分为静态内部类,成员内部类,局部内部类,匿名内部类四种。

1. 静态内部类

public class Out1 {
    private static int a;
    private int b;

    public static class Inner1 {
        public void print() {
            System.out.println(Out1.a);
            System.out.println(new Out1().b);
        }
    }

    public static void main(String[] args) {
        Out1.Inner1 inner1 = new Out1.Inner1();
        inner1.print();
    }

}
  • 静态内部类可以访问外部类所有的静态变量和方法,即使是 private 的也一样。
  • 静态内部类和一般类一致,可以定义静态变量、方法,构造方法等。
  • 其它类使用静态内部类需要使用“外部类.静态内部类”方式,如下所示:Out.Inner inner = new Out.Inner();inner.print();
  • Java 集合类 HashMap 内部就有一个静态内部类 Entry。Entry 是 HashMap 存放元素的抽象,HashMap 内部维护 Entry 数组用了存放元素,但是 Entry 对使用者是透明的。像这种和外部类关系密切的,且不依赖外部类实例的,都可以使用静态内部类。

2. 成员内部类

public class Out2 {
    private static int a;
    private int b;

    public class Inner2 {

        public void print() {
            System.out.println(Out2.a);
            System.out.println(Out2.this.b);
        }
    }

    public static void main(String[] args) {
        Out2.Inner2 inner2 = new Out2().new Inner2();
        inner2.print();
    }
}

定义在类内部的非静态类,就是成员内部类。成员内部类不能定义静态方法和变量(final 修饰的 除外)。这是因为成员内部类是非静态的,类初始化的时候先初始化静态成员,如果允许成员内 部类定义静态变量,那么成员内部类的静态变量初始化顺序是有歧义的。

3. 局部内部类

public class Out3 {
    private static int a;
    private int b;

    public Out3() {
    }

    public void test(final int c) {
        int d = true;

        class Inner3 {
            Inner3(final int val$c) {
                this.val$c = var2;
            }

            public void print() {
                System.out.println(this.val$c);
            }
        }

    }
}

定义在方法中的类,就是局部类。如果一个类只在某个方法中使用,则可以考虑使用局部类。

4. 匿名内部类

interface Print {
    void print();
}
public class Out4 {
    private static int a;
    private int b;

    public void print() {
        new Print() {
            @Override
            public void print() {
                System.out.println(Out4.a);
                System.out.println(Out4.this.b);
            }
        }.print();
    }
}
  • 要继承一个父类或者实现一个接口、直接使用 new 来生成一个对象的引用
  • 匿名内部类我们必须要继承一个父类或者实现一个接口,当然也仅能只继承一个父类或者实现一个接口。同时它也是没有 class 关键字,这是因为匿名内部类是直接使用 new 来生成一个对象的引

看一个案例

public class DotThis {
    void f() {
        System.out.println("DotThis.f()");
    }

    public class Inner {
        public DotThis outer() {
            return DotThis.this;
        }
    }

    public Inner inner() {
        return new Inner();
    }

    public static void main(String[] args) {
        DotThis dt = new DotThis();
        DotThis.Inner dti = dt.inner();
        dti.outer().f();
    }
}