1.为什么使用内部类?

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


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

  • 内部类可以用多个实例,每个实例都有自己的状态信息,并且与其他外围对象的信息相互独立
  • 在单个外围类中,可以让多个内部类以不同的方式实现同一个接口,或者继承同一个类
  • 创建内类对象的时候并不依赖于外围对象的创建
  • 内部类并没有令人迷惑的“is-a”(is-a代表继承关系,has-a代表从属关系),它就是一个独立的主体。
  • 内部类提供了更好的封装,除了该外围类,其他类都不能访问。

2.内部类的分类

1.成员内部类:
  • Inner 类中定义的show()方法可以直接访问 outer类的一个成员变量位置,Inner类可以使用任意访问控制符,如:public,protected,private等
  • Inner类中定义的show()方法可以直接访问 outer 类中的数据,而不受访问控制符的影响,比如可以直接访问 Outer 类中的私有属性 age
  • 定义了成员内部类后,必须使用外部类对象来创建内部类对象,而不能直接去 new 一个内部类,即:内部类 对象名 = 外部类对象.new 内部类();
  • 编译包含内部类的的程序后,会产生两个 .class 文件,Outer.class;Outer$Inner.class
  • 成员内部类不能存在任何static 的变量和方法,可以定义常量:
    • 原因:非静态内部类是依赖于外部类的实例,而静态变量和方法是不依赖对象的,仅与类相关
    • 在加载静态域时,根本没有外部类,所在非静态内部类中定义静态域或方法,否则编译不通过;非静态内部类的作用域是实例级别。
    • 常量是在编译器就确定的,放到所谓的常量池了。

    注意:
    1.外部类是不能直接使用外部类的成员和方法的,可先创建内部类的对象,然后通过内部类的对象来访问其成员和方法。
    2.如果外部类和内部类具有相同的成员变量或方法,内部类默认访问自己的成员变量或方法,如果要访问外部的成员变量,可以使用 this 关键字,如:Outer.this.name
2.静态内部类:即 static 修饰的内部类
  1. 静态内部类不能直接访问外部类的非静态成员,但可以通过 new Outer().成员 ,的方式访问
  2. 如果外部类的静态成员与内部类的成员名称相同,可通过 “类名.静态成员”访问访问外部类的静态成员;如果外部类的静态成员与内部类的成员名称不相同,可通过“成员名”直接调用外部类的静态成员。
  3. **创建静态内部类的对象时,**不需要外部类的对象,可以直接创建 内部类 对象名 = new 内部类();
3.方法内部类:访问权限与方法内或者该作用域内
  1. 方法内部类就像是方法里面的一个局部变量一样,是不能有 public/protected/private /static修饰符的
  2. 只能访问方法中定义的final类型的局部变量,因为:
    • 当方法被调用运行完毕后,局部变量就消亡了。但内部类对象可能还存在,直到没有被引用时才会消亡。此时就会出现一种情况;就是内部类要访问一个不存在的局部变量;
    • 使用final修饰符不仅会保持对象的引用不会被改变,而且编译器还会持续维护这个对象在回调方法中的生命周期。
    • 局部内部类并不是直接调用方法传进来的参数,而是内部类将传进来的参数通过自己的构造器备份到了自己的内部。自己内部的方法调用的实际是自己的属性;而导致内部类得到的值不一致

    使用的形参 为何 要为 final??

在内部类的属性和外部方法的参数两者从外表上看是同一个东西,但实际上却不是,所以他们两者是可以任意变化的,也就是说在内部类中,对属性的改变并不会影响到外部的形参,然而这从程序的角度来看这是不可行的,毕竟站在程序的角度来看这根本就是同一个,如果内部类改变了,而外部方法的形参却没有改变,这是难以理解和不可接受的,所以为了保持参数的一致性,就规定使用 final 来避免形参的不改变

4.匿名内部类
  1. 匿名内部类是直接使用 new 来生成一个对象的引用;
  2. 对于匿名内部类的使用它是存在一个缺陷的,就是它仅仅能被使用一次,创建匿名内部类时它会立即创建一个该类的实例,该类的定义会立刻消失,所以匿名内部类是不能够被重复使用
  3. 使用匿名内部类时,我们必须继承一个类或者实现一个接口,但是两者不可兼得,同时只能继承一个类或实现一个接口。
  4. 匿名内部类中是不能定义构造函数的,匿名内部类中不能存在任何静态成员变量和静态方法,匿名内部类不能是抽象的,它必须要实现继承类,或者实现接口的所有抽象方法。
  5. 匿名内部类初始化:使用构造代码块,利用构造代码块能够达到为匿名内部类创建一个构造器的效果。

更多可参考他人博客学习:https://www.cnblogs.com/chenssy/p/3388487.html