概述:
- 当一个事物的内部,还有一个部分需要一个完整的结构进行描述,而这个内部的完整的结构又只为外部事物提供服务,那么整个内部的完整结构最好使 用内部类。
- 在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
-
匿名内部类
匿名内部类不能定义任何静态成员、方法和类,只能创建匿名内部类的一个实例。一个匿名内部类一定是在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显卡”;
}
});
}
}