1、简介
内部类(inner class)是定义在另一个类中的类。使用内部类的原因,有以下三点:
- 内部类方法可以访问可以访问该类定义所在的作用域中的数据,包括私有的数据。
- 内部类可以对同一个包中的其他类隐藏起来。
- 当想要定义一个回调函数时且不想编写大量代码时,使用匿名内部类比较便捷。
2 、使用内部类访问对象状态
下面举个语音闹钟的例子来说明内部类的使用方式。构造一个语音时钟需要提供两个参数:发布通告的间隔和开关铃声的标志。
public class TalkingClock{
private int interval;
private boolean beep;
public talkingClock(int interval,boolean beep){
this.interval=interbal;
this.beep=beep;
}
public void start(){
ActionListener listener=new TimePrinter();
Timer t=new Timer(interval,listener);
t.start();
}
public class TimePrinter implements ActionListener{
//inner class 内部类
@Override
public void actionPerformed(ActionEvent event){
Date now=new Date();
System.out.println("the time is "+now);
if(!beep) Toolkit.getDefaultToolkit().beep();
}
}
}
public interface ActionListener{
void actionPerformed(ActionEvent event);
}
可以发现,TimePrinter类没有实例域或者名为beep的变量,取而代之的是beep引用了创建了TimePrinter的TalkingClock对象的域。内部类既可以访问自身的数据域,也可以访问创建它的外围类对象的数据域。
如果想实现这种调用,总要有一个隐式引用吧,它指向了创建它的外部类对象。如下图所示:
然而这个引用在内部类的定义是不可见的。java编译器修改了所有内部类的构造器,添加了一个外围类的引用的参数。因为TimePrinter中没有构造器,所以编译器就为其生成了一个默认构造器,如下所示:
public TimePrinter(TalkingClock clock){
outer=clock;
}
outer就是一个例子,并不是java关键字。
3、内部类的特殊语法规则
上一部分对于beep的引用不是非常正规,正规的语法更麻烦些。表达式 OuterClass.this表示外围类的引用。例如可以像下面这样写:
TalkingClock.this.beep
反过来,可以采用下列语法格式更加明确地编写 内部对象的构造器:
outerClss.new InnerClass(constructor parameter)
例如:
ActionListener listener=this.new TimePrinter();
在这里,最新构造地TimePrinter对象的外围类引用被设置为创建内部类对象的方法中的this引用。这是一种很常见的情况。通常,this限定词是多余的。不过,可以通过显式地命名将外围类引用设置成其他对象。例如,TimerPrinter是一个公有的内部类,对于任意的语音时钟都可以构造成一个TimePrinter:
TalkingClock jabberer =new TalkingClock(1000,true);
TalkingClock.TimePrinter listener=jabberer.new TimePrinter();
4、局部内部类
TimePrinter这个类只在start方法中使用了一次。因此,可以使用内部类
public void start(){
class TimePrinter implements ActionListener{
//inner class 内部类
@Override
public void actionPerformed(ActionEvent event){
Date now=new Date();
System.out.println("the time is "+now);
if(!beep) Toolkit.getDefaultToolkit().beep();
}
}
ActionListener listener=new TimePrinter();
Timer t=new Timer(interval,listener);
t.start();
}
局部类不能用public或private访问说明符进行声明。它的作用被限定在这个声明这个局部类的块中。
局部类有一个优势,即对外部世界完全隐藏起来。即使TalingClock中的其他代码也不能访问它。除start方法之外,没有任何方法知道TimePrinter的存在。