Java支持哪些嵌套类?
Java将嵌套类分成两个主要的分类嵌套顶级类(nested toplevel classes)和内部类(inner classes)Java还进一步将内部类分为实例内部类(instance inner class) 本地内部类(local inner class)和匿名内部类(and anonymous inner class)要掌握嵌套类你需要懂得每个分类以下我们就来讨论以下这些分类首先我们来看一下嵌套顶级类
嵌套顶级类
当你在其它类的外面声明一个类时Java就认为该类是一个顶级类如果你在一个顶级类中声明一个类并且在该嵌套类的声明前加上static的修饰符你就得到了一个嵌套顶级类以下的代码段声明了一个顶级类和一个嵌套顶级类
class TopLevelClass
{
static class NestedTopLevelClass
{
}
}
就象静态的字段和静态的方法一样(也可称为类字段和类方法)一个嵌套的顶级类也是与对象无关的考虑以下的代码段
class TopLevelClass
{
static int staticField;
int instanceField;
static class NestedTopLevelClass
{
static
{
Systemoutprintln (Can access staticField + staticField);
// Systemoutprintln (Cannot access instanceField + instanceField);
}
{
Systemoutprintln (Can access staticField + staticField);
// Systemoutprintln (Cannot access instanceField + instanceField);
}
}
}
在上面NestedTopLevelClass类的类初始化和对象初始化代码块中你可以访问TopLevelClass的staticField变量不过你在这两个代码块中都不能访问instanceField变量因为NestedTopLevelClass和任何的TopLevelClass对象都是无关的所以NestedTopLevelClass 不能访问TopLevelClass的instanceField变量
警告一个嵌套顶级类不能访问任何外部类的实例成员(包括字段和方法)
虽然NestedTopLevelClass不能访问TopLevelClass的实例字段但是static的修饰符并不会妨碍NestedTopLevelClass声明自己的实例字段以及创建NestedTopLevelClass对象列表的代码就是一个很好的例子
Listing NestedTopLevelClassDemojava
// NestedTopLevelClassDemojava
class TopLevelClass
{
static class NestedTopLevelClass
{
int myInstanceField;
NestedTopLevelClass (int i)
{
myInstanceField = i;
}
}
}
class NestedTopLevelClassDemo
{
public static void main (String [] args)
{
TopLevelClassNestedTopLevelClass ntlc;
ntlc = new TopLevelClassNestedTopLevelClass ();
Systemoutprintln (ntlcmyInstanceField);
}
}
在运行后NestedTopLevelClassDemo产生以下的输出
NestedTopLevelClassDemo的main()方法创建了一个NestedTopLevelClass变量ntlc声明该变量的语法和列表的一样(EmployeeJobIterator eji = egetJobIterator ()) 通常当你需要一个嵌套类的变量时要在该嵌套类名的前面加入所有外部类的名字并且用将它们分开用new关键字声明该嵌套类的时候也要这样做
这时你可能想知道你是否可以在一个嵌套顶级类中声明一个顶级嵌套类还有如果两个不同的外部类声明了同样的字段变量名但该变量的类型或者初始化值是不同的?要解答这些问题可看列表
Listing NestingAndShadowingDemojava
// NestingAndShadowingDemojava
class TopLevelClass
{
private static int a = ;
private static int b = ;
static class NestedTopLevelClass
{
private static int a = ;
static class NestedNestedTopLevelClass
{
void printFields ()
{
Systemoutprintln (a = + a);
Systemoutprintln (b = + b);
}
}
}
}
class NestingAndShadowingDemo
{
public static void main (String [] args)
{
TopLevelClassNestedTopLevelClassNestedNestedTopLevelClass nntlc;
nntlc = new TopLevelClassNestedTopLevelClass
NestedNestedTopLevelClass ();
nntlcprintFields ();
}
}
在运行时NestingAndShadowingDemo产生以下的输出
a =
b =
NestingAndShadowingDemo可以编译运行证明你可以在顶级嵌套类中嵌套顶级嵌套类结果也显示了NestedTopLevelClass的一个字段屏蔽了TopLevelClass的字段结果是打印出NestedTopLevelClass的字段内容
使用嵌套顶级类你不可以访问外部类的实例字段或者调用该类的实例方法要访问实例成员Java支持内部类内部类和嵌套顶级类相似不同的地方是你不会在内部内的声明前加上static关键字我们以下就讨论一下内部类先从实例内部类开始
提示你可以在顶级嵌套类前加上privateprotected或者public关键字以指示该类在外部类以外的访问级别
实例内部类(Instance inner classes)
假设你在声明嵌套类时没有在前面加上static关键字这样你得到的不是一个嵌套顶级类你得到的是一个实例内部类嵌套顶级类只可以访问外部内的static成员而实例内部类可以同时访问static和实例成员列表的代码中有实例内部内的例子在Employee类中你可以看到实例内部类JobIterator仔细观察这两个类你可以看到JobIterator可以访问Employee的private jobs实例字段
提示你可以在实例内部类前加上privateprotected或者public关键字以指示该类在外部类以外的访问级别
本地内部类(Local inner classes)
除了以上的类嵌套外Java还允许你在任意的代码段中放入一个类该类的代码放在一对大括号中({ })这意味着类可以出现在一个方法中甚至是在if语句的一对括号中这样的类就称为本地内部类
相对于实例内部类来说本地内部类与有一个好处它除了可以访问外部类的实例和类字段外(称为实例和类方法)还可以访问本地变量和方法的参数列表6就是一个本地内部类
Listing LocalInnerClassDemojava
// LocalInnerClassDemojava
import javautil*;
class ComputerLanguage
{
private String name;
ComputerLanguage (String name)
{
thisname = name;
}
public String toString ()
{
return name;
}
}
class LocalInnerClassDemo
{
public static void main (String [] args)
{
ComputerLanguage [] cl =
{
new ComputerLanguage (Ada)
new ComputerLanguage (Algol)
new ComputerLanguage (APL)
new ComputerLanguage (Assembly IBM )
new ComputerLanguage (Assembly Intel)
new ComputerLanguage (Assembly Mostek)
new ComputerLanguage (Assembly Motorola)
new ComputerLanguage (Assembly VAX)
new ComputerLanguage (Assembly Zilog)
new ComputerLanguage (BASIC)
new ComputerLanguage (C)
new ComputerLanguage (C++)
new ComputerLanguage (Cobol)
new ComputerLanguage (Forth)
new ComputerLanguage (Fortran)
new ComputerLanguage (Java)
new ComputerLanguage (LISP)
new ComputerLanguage (Logo)
new ComputerLanguage (Modula )
new ComputerLanguage (Pascal)
new ComputerLanguage (Perl)
new ComputerLanguage (Prolog)
new ComputerLanguage (Snobol)
};
Enumeration e = enumerator ((Object []) cl);
while (ehasMoreElements ())
Systemoutprintln (enextElement ());
}
static Enumeration enumerator (final Object [] array)
{
class LocalInnerClass implements Enumeration
{
private int index = ;
public boolean hasMoreElements ()
{
return index < array.length;
}
public Object nextElement ()
{
return array [index++].toString ();
}
}
return new LocalInnerClass ();
}
}
在运行后, LocalInnerClassDemo产生以下的输出:
Ada
Algol
APL
Assembly - IBM 360
Assembly - Intel
Assembly - Mostek
Assembly - Motorola
Assembly - VAX
Assembly - Zilog
BASIC
C
C++
Cobol
Forth
Fortran
Java
LISP
Logo
Modula 2
Pascal
Perl
Prolog
Snobol
LocalInnerClassDemo展示了在LocalInnerClassDemo的enumerator()类方法中声