Java 中的内部类

1.为什么使用内部类

使用内部类最吸引人的原因是:每个内部类都能独立地实现一个接口,所以无论外围类是否已经实现了某个接口,对于内部类都没有影响

1.1使用内部类最大的优点就在于它能够非常好的解决多重继承的问题,使用内部类还可以为我们带来如下特性:

  • 内部类可以用多个实例,每个实例都有自己的状态信息,并且与其他外围对象的信息相互独立

  • 在某个外围类中,可以让多个内部类 以不同的方式实现同一个接口,或者继承同一个类

  • 创建内部类对象的时刻并不依赖于外部类对象的创建

  • 内部类提供了更好的封装,除了外围类,其他类都不能访问

2.内部类分类

2.1成员内部类

  • Inner 类定义在 outer 类的内部,相当于 outer 类的一个成员变量的位置,Inner 类可以使用任意访问控制符

  • Inner 类中定义的 show() 方法可以直接访问 outer 类中的数据,而不受访问控制符的影响

  • 定义成员内部类后,必须使用外部类 对象来构建内部类对象,而不能直接去 new 一个内部类对象

    • 即:内部类 对象名 = 外部类实例.new 内部类()
  • 编译包含内部类的程序后会包含两个 .class 文件:outer.class,outer$Inner.class

  • 成员内部类中不能存在任何 static 的变量和方法,可以定义常量

    • 因为非静态内部类是要依赖于外部类的实例,而静态对象和方法是不依赖于对象的,仅与类相关

    • 简而言之:在加载静态域时,根本没有外部类,所以在非静态内部类中不能定义静态域或方法,否则编译不通过;非静态内部类的作用域是实例级别

    • 常量是在编译器就确定的,放到所谓的常量池了

  • 注意:

    • 外部类是不能直接使用内部类的成员和方法的,可先创建内部类的对象,然后通过内部类的对象来访问其成员变量和方法

    • 如果外部类和内部类具有相同的成员变量或方法,内部类默认访问自己的成员变量或方法,如果要访问外部类的成员变量,可以使用 this关键字,如: outer.this.name

2.2静态内部类:是 static 修饰的内部类

  • 静态内部类不能直接访问外部类的非静态成员,但可以通过 new 外部类().成员 的方式访问

  • 如果外部类的静态成员和内部类的成员名称相同,可通过 类名.静态成员 访问外部类的静态成员

  • 如果外部类的静态成员和内部类的成员名称不同,则可通过成员名直接调用外部类的静态成员

  • 创建静态内部类的对象时,不需要外部类的对象,可以直接创建 内部类 对象名 = new 内部类()

2.3方法内部类:访问仅限于方法内或者该作用域内

  • 方法内部类就像是方法里面的一个局部变量一样,是不能有 public、protected、private 以及 static 修饰符的

  • 只能访问方法中定义的 final 类型的局部变量,因为:当方法被调用完成后局部变量就已经消亡了。但内部类对象可能还存在,直到没有被引用时才会消亡。此时就会出现一种情况,就是内部类要访问一个不存在的局部变量。

    • 使用 final 修饰符不仅会保持对象的引用不会改变,而且编译器还会持续维护这个对象在回调方法中的生命周期

    • 方法内部类并不是直接调用方法传进来的参数,而是内部类将传进来的参数通过自己的构造器备份到了自己内部,方法内部类调用的实际上是自身的属性而不是外部类方法的参数

2.4匿名内部类

  • 匿名内部类是直接使用 new 来生成一个对象的引用

  • 对于匿名内部类的使用是存在缺陷的,就是它仅能被使用一次,创建匿名内部类时它会立即创建一个该类的实例,该类的定义会立即消失,所以匿名内部类是不能够被重复使用

  • 使用匿名内部类时,我们必须是继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或实现一个接口

  • 匿名内部类中是不能定义构造函数的,匿名内部类中不能存在任何的静态成员变量和静态方法

  • 匿名内部类不能是抽象的,他必须要实现继承的类或者实现接口的所有抽象方法

-----------本文结束感谢您的阅读-----------
0%