内部类:
定义在其他类内部的类叫做内部类(嵌套类),包含内部类的类称为外部类(宿主类)。
a)内部类提供更好的封装,不允许同一包中的其他类访问该类。
b)内部类可直接访问外部类的私有数据(同一类的成员可互相访问),但外部类不能访问内部类的实现细节。
c)匿名内部类适合用于创建那些仅需要一次使用的类。
1、非静态内部类
成员内部类是与Field(属性)、方法、构造器、初始化块相似的类成员。局部内部类和匿名内部类则不是类成员。
成员内部类包括:1、静态内部类---->用static修饰的成员内部类;2、非静态内部类---->没用static修饰的成员内部类。
成员内部类的class文件形式:OuterClass$InnerClass.class
调用原理:在非静态内部类的方法内访问某个变量时,
a)系统优先在该方法内查找变量是否存在
b)在到方法所在的内部类中查找
c)在到内部类所在的外部类中查找,若仍不在则编译出错。
因此当外部类成员变量、内部类成员变量、内部类方法局部变量同名时,则可通过用this(访问非静态内部类的成员变量)、外部类.this(外部类成员变量)来限定区分。
外部类以外的地方访问非静态内部类时:
Outer.Inner in=new Outer().new Inner();
不允许在外部类的静态成员中使用非静态内部类。
非静态内部类不能定义静态成员(静态方法、静态属性(Field)、静态初始化块),但可以包含普通初始化块。
2、静态内部类
用static修饰的成员内部类称为静态内部类,其属于外部类本身,被称为类内部类。
【static关键字是把类的成员变成类相关,而不是实例相关,即static修饰的成员属于整个类。static不可修饰外部类,但可修饰内部类。因为外部类上一级程序单元是包,故不可以用static;而内部类上一级程序单元是外部类,故使用static则变成外部类相关,而不是外部类实例相关】
规则:静态成员不能访问非静态成员。
静态内部类可包含:静态成员和非静态成员。
静态内部类不能访问外部类的实例成员,只能访问外部类的类成员,静态内部类实例方法也不能访问外部类的实例成员,只能访问外部类的静态成员。
外部类不能直接访问静态内部类成员,但可用:1)用静态类名调用静态内部类的类成员(静态成员)。 2)用静态内部类对象(即new Inner().)访问静态实例变量。
接口中内部类默认public static修饰,即接口内部类只能是静态内部类。
外部类以外的地方访问静态内部类时:
Outer.Inner in=new Outer.Inner();
【注意】
a)不要在外部类的静态成员(静态方法、静态初始化块)中使用非静态内部类。
b)在外部类以外的地方访问内部类(包括静态和非静态两种),则内部类不能用private访问控制权限,private修饰的只能在外部类内部使用。
3、局部内部类
把一个内部类放在方法里对定义,则此类叫局部内部类。局部内部类只能在该方法里有效,故其不能在外部类的方法以外的地方使用,因此其也不能使用访问控制符和static修饰符。
static修饰的成员就是类成员。
【对于局部成员而言,不管是局部变量还是局部内部类,它的上一级程序单元都是方法,而不是类,使用static修饰没有任何意义;因为局部成员的作用域是其所在方法,其他程序单元永远也不可能访问另一方法中的局部成员,所以所有局部成员都不能使用访问控制符,也不能使用static修饰。】
若要用局部内部类定义变量、创建实例或派生子类,则只能在其所在方法内进行。
4、匿名内部类
匿名内部类适合创建那种只需要一次使用的类。成创建匿名内部类时系统会立即创建一个该类的实例,这个类定义立即消失,且匿名内部类不能重复使用。
格式:new 父类()/接口() { //匿名内部类类体 }
如:
interface Product{
public String getName();
public double getPrice();
}
public class AnonymousTest {
public void test(Product p){
System.out.println("购买一本"+p.getName()+"花掉"+p.getPrice());
}
public static void main(String[] args) {
AnonymousTest at=new AnonymousTest();
//调用test方法时,需要传入一个Product参数
//此处传入其匿名内部类的实例
at.test(new Product(){
public String getName(){
return "《Android开发》";
}
public double getPrice(){
return 38.8;
}
);
}
}
由上,匿名内部类必须继承一个父类或实现一个接口,但最多只能继承一个父类,或实现一个接口。
【注意】
A)匿名内部类不能是抽象类,因为系统在创建匿名内部类时,会立即创建匿名内部类的对象(抽象类不能被实例化)。
B)匿名内部类不能定义构造方法(构造器),因为匿名内部类没有类名,所以无法定义构造方法,但匿名内部类可以定义实例初始化块,通过实例初始化块完成构造方法需要完成的事。
C)匿名内部类若访问外部类的局部变量,则局部变量必须是final修饰的。