一、概述
定义在其他类内部的类,被称为内部类(或嵌套类);包含内部类的类被称为外部类(或宿主类)。
- 内部类会
延迟加载
,只有当第一次使用的时候才会被加载,不使用就不胡被加载。 - 内部类提供了更好的封装性,可以把内部类隐藏在外部类之内,不允许同一个包下的其他类访问。
- 内部类可以直接访问外部类的私有数据,因为
内部类被当作其外部类的成员
,同一个类的成员之间可以互相访问。但外部类不能访问内部类的实现细节。 - 内部类比外部类多了三个修饰符:private、protected、static。外部类不可以使用这三个修饰符。
- 非静态内部类不能拥有静态成员。
大部分时候,内部类都被作为成员内部类定义,而不是作为局部内部类。成员内部类是一种于成员变量、方法、构造器和初始化块相似的成员;局部内部类和匿名内部类则不是类成员。
二 非静态内部类
2.1 非静态内部类与外部类的关系
- 非静态内部类的成员可以访问外部类的private成员,但是外部类不能访问非静态内部类的成员。
- 非静态内部类对象必须寄生在外部类对象中,而外部类对象不一定有非静态内部类对象寄生在其中。
在非静态内部类中,可以直接访问外部类的private实例变量。这是因为在非静态内部类对象中,保存了一个它所寄生的外部类对象的引用。
因为当调用非静态内部类的实例方法时,必须有一个非静态内部类实例,非静态内部类实例必须寄生在其外部类实例中。非静态内部类的创建需要依赖于外部类:
Outer outer = new Outer();
Inner inner = outer.new Inner();
2.2 非静态内部类访问外部类
如果在外部类、内部类、内部类方法中都定义了一个同名的变量,当在非静态内部类的方法中访问这个变量时,系统会依次寻找:
- 该方法内的局部变量。
- 该方法所在内部类的成员变量。
- 内部类所在的外部类的成员变量。
但是也可通过使用this、外部类类名.this加以区分:
- 外部类中的成员变量:
外部类类名.this.变量名
。 - 内部类中的成员变量:
this.变量名
。 - 方法中的局部变量:
变量名
。
public class Outer {
public String msg = "Outer msg";
class Inner{
public String msg = "Inner msg";
public void printMsg(){
String msg = "Inner Method msg";
System.out.println(msg); // Inner Method msg
System.out.println(this.msg); // Inner msg
System.out.println(Outer.this.msg); // Outer msg
}
}
public static void main(String[] args) {
Outer outer = new Outer();
Inner inner = outer.new Inner();
inner.printMsg();
}
}
2.3 外部类访问非静态内部类
非静态内部类的成员可以访问外部类的private成员,但是非静态内部类的成员只有在非静态内部类范围内是可知的,并不能被外部类直接使用。
如果外部类需要访问非静态内部类的成员,则必须 显示创建非静态内部类对象 来调用其实例成员。例如下来代码,在外部类的方法中访问非静态内部类的成员:
public void printInnerMsg(){
System.out.println(new Inner().msg);
}
三、静态内部类
使用static修饰一个内部类,则这个内部类就属于外部类本身,而不属于外部类的某个对象。
静态内部类的方法仍不能访问外部类的实例成员。因为静态内部类是外部类相关的,而不是外部类对象相关的。即静态内部类对象不是寄生在外部类的实例中,而是寄生在外部类本身当中。静态内部类只持有外部类的类引用,而不持有外部类对象的引用。静态内部类存在,不一定存在外部类对象。
四、局部内部类
如果把一个内部类定义在一个方法中,则这个内部类就是一个局部内部类。
五、匿名内部类(重点)
匿名内部类适合创建那种只需一次使用的类,创建匿名内部类时会创建一个该类的实例,这个类定义立即消失,匿名内部类不能重复使用。
new 实现接口() | 父类构造器(实参列表)
{
// 匿名内部类的类体
}
- 匿名内部类必须继承一个父类,或实现一个接口,但最多只能继承一个父类,实现一个接口。
- 匿名内部类不能是抽象类,因为系统在创建匿名内部类时,会立即创建匿名内部类的对象。
- 匿名内部类不能定义构造器。由于匿名内部类没有类名,所有无法定义构造器。但匿名内部类可以定义初始化块。
- 创建匿名内部类时,必须实现接口或抽象父类中的所有抽象方法。
- 被局部内部类和匿名内部类访问过的局部变量,会自动被final修饰。