概述:

  • 当一个事物的内部,还有一个部分需要一个完整的结构进行描述,而这个内部的完整的结构又只为外部事物提供服务,那么整个内部的完整结构最好使 用内部类。
  • 在Java中,允许一个类的定义位于另一个类的内部,前者称为内部类(Inner class),后者称为外部类。

内部类的分类:

成员内部类(static成员内部类和非static成员内部类)

  • 成员内部类作为类的成员的角色:
    • 和外部类不同,内部还可以声明为private或protected;
    • 可以调用外部类的结构
    • Inner class 可以声明为static的,但此时就不能再使用外层类的非static的成员变量;
  • 成员内部类作为类的角色:
    • 可以在内部定义属性、方法、构造器等结构
    • 可以声明为abstract类 ,因此可以被其它的内部类继承
    • 可以声明为final的
    • 编译以后生成OuterClass$InnerClass.class字节码文件(也适用于局部内部类)
  • 在其他类中创建成员内部类的对象(静态的,非静态的):
    • 创建静态的Dog内部类的实例(静态的成员内部类):

Person.Dog dog = new Person.Dog();

  • 创建非静态的Bird内部类的实例(非静态的成员内部类):

//Person.Bird bird = new Person.Bird();//错误的
Person p = new Person();——->先创建外部类对象,因为非静态内部类随着外部类对象的创建而加载
Person.Bird bird = p.new Bird();

  • 在成员内部类中调用外部类的结构

class Person{
String name = “小明”;
public void eat(){
}
public static void marry(){
}

//非静态成员内部类
class Bird{
String name = “杜鹃”;
public void display(String name){
System.out.println(name);//方法的形参
System.out.println(this.name);//内部类的属性
System.out.println(Person.this.name);//外部类的属性
//eat() 等价于 Person.this.eat(); ————>调用外部类的非静态方法
//marry(); 等价于 Person.marry(); ————>调用外部类的静态方法

}
}
}

  • 注:
    • 非static的成员内部类中的成员不能声明为static的,只有在外部类或static的成员内部类中才可声明static成员。
    • 外部类访问自己的成员内部类的成员,需要“内部类.成员”(静态方法或属性)或“内部类对象.成员”(非静态方法或属性)的方式
    • 成员内部类可以直接使用外部类的所有成员,包括私有的数据 ,但静态内部类中不可调用外部类中非静态的结构;—->生命周期理解
    • 当想要在外部类的静态成员部分使用内部类时,可以考虑内部类声明为静态的,外部类中的静态成员和其静态内部类同时加载;

局部内部类(不谈修饰符)

  • 定义在方法内、代码块内、构造器内的类叫做局部内部类;
  • 如何使用局部内部类
    • 只能在声明它的方法或代码块中使用,而且是先声明后使用。除此之外的任何地方都不能使用该类
    • 但是它的对象可以通过外部方法的返回值返回使用,返回值类型只能是局部内部类的父类或父接口类型;

例:返回一个实现了Comparable接口的类的对象
public Comparable getComparable(){

//创建一个实现了Comparable接口的类:局部内部类
//方式一:
class MyComparable implements Comparable{

@Override———>实现Comparable接口
public int compareTo(Object o) {
return 0;
}
}

return new MyComparable();——>返回一个内部类的对象

//方式二:
return new Comparable(){———->创建接口的匿名实现类的匿名对象

@Override———>实现Comparable接口
public int compareTo(Object o) {
return 0;
}

};

}

  • 局部内部类的特点
    • 内部类仍然是一个独立的类,在编译之后内部类会被编译成独立的.class文件,但是前面冠以外部类的类名和$符号,以及数字编号。
    • 只能在声明它的方法或代码块中使用,而且是先声明后使用。除此之外的任何地方 都不能使用该类。
    • 局部内部类可以使用外部类的成员,包括私有的。
    • 方法内的局部内部类可以使用外部方法的局部变量,但是此属性必须是final的。。由局部内部类和局部变量的声明周期不同所致。

jdk 7及之前版本:要求此局部变量显式的声明为final的
jdk 8及之后的版本:可以省略final的声明(相当于传了一个final副本)

  • 局部内部类和局部变量地位类似,不能使用public,protected,缺省,private
  • 局部内部类不能使用static修饰,因此也不能包含静态成员

    匿名内部类

  • 匿名内部类不能定义任何静态成员、方法和类,只能创建匿名内部类的一个实例。一个匿名内部类一定是在new的后面,用其隐含实现一个接口或实现一个类。

  • 格式:

new 父类构造器(实参列表)|实现接口(){
匿名内部类的类体部分
}

  • 匿名内部类的特点
    • 匿名内部类必须继承父类或实现接口
    • 匿名内部类只能有一个对象
    • 匿名内部类对象只能使用多态形式引用
  • 例:

interface Product{
public double getPrice();
public String getName();
}
public class AnonymousTest{
public void test(Product p){只能使用多态形式引用
System.out.println(“购买了一个” + p.getName() + “,花掉了” + p.getPrice());
}
public static void main(String[] args) {
AnonymousTest ta = new AnonymousTest();
//调用test方法时,需要传入一个Product参数,
//此处传入其匿名实现类的实例
ta.test(new Product(){
public double getPrice(){
return 567.8;
}
public String getName(){
return “AGP显卡”;
}
});
}
}