Java内部类
软件编程
228
皇太极
: 2021-04-25 18:25:42

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

1. 静态内部类

  1. public class Out1 {
  2. private static int a;
  3. private int b;
  4. public static class Inner1 {
  5. public void print() {
  6. System.out.println(Out1.a);
  7. System.out.println(new Out1().b);
  8. }
  9. }
  10. public static void main(String[] args) {
  11. Out1.Inner1 inner1 = new Out1.Inner1();
  12. inner1.print();
  13. }
  14. }
  • 静态内部类可以访问外部类所有的静态变量和方法,即使是 private 的也一样。
  • 静态内部类和一般类一致,可以定义静态变量、方法,构造方法等。
  • 其它类使用静态内部类需要使用“外部类.静态内部类”方式,如下所示:Out.Inner inner = new Out.Inner();inner.print();
  • Java 集合类 HashMap 内部就有一个静态内部类 Entry。Entry 是 HashMap 存放元素的抽象,HashMap 内部维护 Entry 数组用了存放元素,但是 Entry 对使用者是透明的。像这种和外部类关系密切的,且不依赖外部类实例的,都可以使用静态内部类。

2. 成员内部类

  1. public class Out2 {
  2. private static int a;
  3. private int b;
  4. public class Inner2 {
  5. public void print() {
  6. System.out.println(Out2.a);
  7. System.out.println(Out2.this.b);
  8. }
  9. }
  10. public static void main(String[] args) {
  11. Out2.Inner2 inner2 = new Out2().new Inner2();
  12. inner2.print();
  13. }
  14. }

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

3. 局部内部类

  1. public class Out3 {
  2. private static int a;
  3. private int b;
  4. public Out3() {
  5. }
  6. public void test(final int c) {
  7. int d = true;
  8. class Inner3 {
  9. Inner3(final int val$c) {
  10. this.val$c = var2;
  11. }
  12. public void print() {
  13. System.out.println(this.val$c);
  14. }
  15. }
  16. }
  17. }

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

4. 匿名内部类

  1. interface Print {
  2. void print();
  3. }
  4. public class Out4 {
  5. private static int a;
  6. private int b;
  7. public void print() {
  8. new Print() {
  9. @Override
  10. public void print() {
  11. System.out.println(Out4.a);
  12. System.out.println(Out4.this.b);
  13. }
  14. }.print();
  15. }
  16. }
  • 要继承一个父类或者实现一个接口、直接使用 new 来生成一个对象的引用
  • 匿名内部类我们必须要继承一个父类或者实现一个接口,当然也仅能只继承一个父类或者实现一个接口。同时它也是没有 class 关键字,这是因为匿名内部类是直接使用 new 来生成一个对象的引

看一个案例

  1. public class DotThis {
  2. void f() {
  3. System.out.println("DotThis.f()");
  4. }
  5. public class Inner {
  6. public DotThis outer() {
  7. return DotThis.this;
  8. }
  9. }
  10. public Inner inner() {
  11. return new Inner();
  12. }
  13. public static void main(String[] args) {
  14. DotThis dt = new DotThis();
  15. DotThis.Inner dti = dt.inner();
  16. dti.outer().f();
  17. }
  18. }