一、概述
我们知道,Java是一种完全的面向对象的语言,作为对象的灵魂,类的种类是多种多样的。类大致可以分外部类和内部类两种,外部类就是我们通常使用的类,而内部类的使用要比外部类少得多,最常见的是GUI事件侦听器。内部类的应用虽然不多,但是如果能够有效地使用内部类,能达到事半功倍的效果。
二、内部类和嵌套类
要讨论内部类和嵌套类,首先要分清它们两者的区别与联系。
首先,内部类(Inner Classes)和嵌套类(Nested Classes)是指在一个类里面定义的另一个类。其次,无论是内部类还是嵌套类,在编译时都被当作一个独立的整体。对于访问它们的其他对象来说(假如它们对这个类来说是可见的)它们的使用和我们通常用的类是一样的。
但是,内部类和嵌套类的区别在于:
嵌套类是静态的,而内部类不是,也就是说嵌套类的实例化不需要外部类的实例,但是内部类是需要这个实例的。
嵌套类可以任意声明静态成员,内部类不允许声明除了编译时常量以外的任何静态成员。这一限制也适用于静态初始化函数。
嵌套类都是命名的,匿名的类声明不能声明运行时静态成员(不管声明是不是静态的)。 注意:类内部声明的接口都属于嵌套类。
下面我们来分别讨论一下它们的具体行为表现。
三、内部类
内部类是指在一个类里面以非静态形式声明的另一个类。内部类的实例化需要外部实例的存在。一个类可以拥有多个内部类,一个外部实例可以拥有多个内部类的实例。但是内部类有且只有一个外部实例,并且在实例化时就已经指定,无法更改。
1. 内部类的分类
内部类根据访问权限不同可以分为以下两种类型:普通内部类和局部内部类。
普通内部类和局部内部类主要的区别在于作用域和访问权限的不同,普通内部类可以被所有人访问(只要访问控制符允许),而局部内部类的作用域更像一个变量,只能在定义它的函数内部被使用,其他人是无法使用这个类的。而且局部类可以访问定义它的函数中的final变量。
内部类根据声明方式不同又可以分为:命名内部类和匿名内部类。
命名内部类和匿名内部类的区别在于:
首先,命名类可以抽象的。匿名类不能为抽象,事实上,匿名内部类在编译时被隐形地声明为最终的(final)。
其次,命名类声明可以继承一个父类并实现多个接口,匿名类只能有一个父类(或者接口)。
再次,命名类可以被多次实例化,而匿名类只能在定义时被实例化一次。最后,命名类可以声明多个构造函数并控制访问哪一个父类的构造函数,匿名类无法声明构造函数。
下面分别描述这些区别:
2. 普通内部类
普通内部类是指在类成员定义中定义的类,这些类可以拥有访问控制符(public, private等)。如果访问控制符允许,则这些类可以被外面直接应用。普通命名内部类可以声明为一个接口,或者是抽象的。嵌套类其实是特殊的普通内部类,但由于其特殊性,故在下面独立讨论。
声明:
命名类:
class OuterClass{
//Outer class definition
class InnerNamedClass{
//Inner class definition
}
}
匿名类:
//声明匿名类时,可以使用一个类或者接口作为它的父类
class OuterClass{
//Outer class definition
Object unnamedObject = new Object(){
//Inner class definition
}
}
实例化:
命名类:
//外部或静态方法:
OuterClass outerObject = new OuterClass();
OuterClass.InnerNamedClass innerObject = outerObject.new InnerNamedClass();
//内部:
InnerNamedClass innerObject = new InnerNamedClass();
匿名类:
定义时即完成实例化
访问权限:
- 内部实例对外部实例的访问权限为:外部类定义或继承的所有字段
- 外部实例对内部实例的访问权限为:内部类定义或继承的所有字段
- 其他对象对内部实例的访问权限为:若内部类不可见,则只能访问其超类定义的字段;若内部类可见,则可访问内部实例的非私有字段,具体情况与通常的类类似
备注:
内部类的外部实例是在构造时作为参数传给构造函数的,在SUN JDK中,保存外部实例的字段通常被声明为:
final OutterClass this$0;
3. 局部内部类
局部内部类,是指在函数体内声明的类,这种类是局域性的,只在函数内声明后有效,它最大的特点是:可以访问定义它的函数中的final变量。
声明:
命名类:
class OuterClass{
//Outer class definition
void aMethod(){
class InnerNamedClass{
//Inner class definition
}
}
}
匿名类:
class OuterClass{
//Outer class definition
void aMethod(){
Object unnamedObject = new Object(){
//Inner class definition
}
}
}
实例化:
命名类:局部类的作用域仅限定义该类的方法中,无法在外部实例化或访问,实例化方法与通常类相同
匿名类: 定义时即完成实例化
访问权限:
- 内部实例对外部实例的访问权限为: 外部类定义或继承的所有字段;定义该内部类的方法在定义类之前所定义的所有final变量
- 外部实例对内部实例的访问权限为:在定义该内部类的方法中可以访问内部类定义或继承的所有字段,在外部实例的其他字段中只能访问其超类定义的字段
- 其他对象对内部实例的访问权限为: 只能访问其超类定义的字段
备注:
当试图访问一个final变量时,编译器实际上把该变量的值赋值给内部类的一个隐含字段中。编译器会自动地在构造函数中添加相应的参数。在SUN JDK中,这个隐含字段通常被命名为:val$varname
4. 使用内部类的注意事项:
本节包括了一些使用内部类时需要注意的事项,这些事项对所有内部类都适用。
静态成员
首先需要注意的是,内部类不能声明非常量静态成员,任何静态成员的声明都会被当成编译错误,例如:static String a;
访问外部实例
要在一个内部实例中访问外部实例,请使用:OuterClass.this
字段覆盖
在内部类中声明的与外部类同签名的字段将覆盖外部字段,要访问外部字段请使用外部实例对象前缀。
OuterClass.this.field
OuterClass.this.Method();
四、嵌套类
嵌套类(Nested Classes)其实是普通内部类的一种特殊形式。首先它的声明是静态的,这就表示了这个类不需要外部实例,也表示了它不能访问外部类的实例字段。但是相应的,嵌套类可以拥有非常量静态成员。事实上,JDK通常把嵌套类当成一个具有特殊名字的独立类。
另外,嵌套类还拥有一种特殊形式:匿名嵌套类。前面说过,匿名类都是内部类,但是匿名嵌套类是一个特例,从理论上讲,它既不属于嵌套类,也不属于内部类。匿名嵌套类不允许拥有非常量静态成员,但是它也没有外部实例供访问。
声明:
命名类:
class OuterClass{
//Outer class definition
static class StaticInnerNamedClass{
//Inner class definition
}
}
匿名类:
class OuterClass{
//Outer class definition
static Object unnamedObject = new Object(){
//Inner class definition
}
}
实例化:
内部:
StaticInnerNamedClass staticInnerNamedObject = new StaticInnerNamedClass();
外部:
OuterClass.StaticInnerNamedClass staticInnerNamedObject = new OuterClass.StaticInnerNamedClass();
匿名嵌套类在声明时即完成实例化。