Java核心技术第6章(3)

6.4 内部类

    内部类(inner class)是定义在另一个类中的类. 为什么需要使用内部类呢?其主要原因有以下三点:
    内部类方法可以访问该类定义所在的作用域中的数据,包括私有数据.
    内部类可以对同一个包中的其他类隐藏起来.
    当想要定义一个回调函数且不想编写大量代码时,使用匿名(annoymous)内部类比较便捷.

    注释:C++有嵌套类,一个被嵌套的类包含在外围类的作用域内.下面是一个典型的例子,一个链表类定义了一个存储结点的类和一个定义迭代器位置的类.
class LinkedList
{
public:
    class Iterator
    {
    public:
        void insert(int x);
        int erase();
        ...
    };
    ...
private:
    class Link
    {
    public:
        Link* next;
        int data;
    };
    ...
};
    嵌套是一种类之间的关系,而不是对象之间的关系.一个LinkedList对象并不包含Iterator类型或Link类型的子对象.
    嵌套类有两个好处:命名控制和访问控制.由于名字Iterator嵌套在LinkedList类的内部,所以在外部被命名为LinkedList::Iterator,这样就不会与其他名为Iterator的类发生冲突.在Java中这个并不重要,因为Java包已经提供了相同的命名控制,需要注意的是,Link类位于LinkedList类的私有部分,因此,Link对其他的代码均不可见.鉴于此情况,可以将Link的数据域设计为公有的,它仍然是安全的.这些数据域只能被LinkedList类中的方法访问,而不会暴露给其他的代码.在Java中,只有内部类能够实现这样的控制.
    然而,Java内部类还有另外一个功能,这使得它比C++的嵌套类更加丰富,用途更加广泛. 内部类的对象有一个隐式引用,它引用了实例化该内部对象的外围类对象.通过这个指针,可以访问外围类对象的全部状态.

6.4.1   使用内部类访问对象状态

    下面分析TimerTest示例,并抽象出一个TalkingClock类,构造一个语音时钟时需要提供两个参数:发布通告的间隔和开关铃声的标志.
public class TalkingClock
{
    private int interval;
    private boolean beep;

    public TalkingClock(int interval, boolean beep) { ... }
    public void start() { ... }
    public class TimePrinter implements ActionListener
    {
        ...
    }
}
    需要注意,这里的TimePrinter类位于TalkingClock类内部,这并不意味着每个TalkingClock都有一个TimePrinter实例域.
    下面是TimePrinter类的详细部分.需要注意一点,actionPerformed方法在发出铃声之前检查了beep标志.
public class TimePrinter implements ActionListener
{
    public vooid actionPerformed(ActionEvent event)
    {
        Date now = new Date();
        System.out.println("At the tone, the time is " + now);
        if (beep)
            Toolkit.getDefaultToolkit().beep();
    }
}
    令人惊讶的事情发生了,TimePrinter类没有实例域或者名为beep的变量,取而代之的是beep引用了创建TimePrinter的TalkingClock对象的域.这是一种创新的想法.从传统意义上将,一个方法可以引用调用这个方法的对象数据域.内部类既可以访问自身的数据域,也可以访问创建它的外围类对象的数据域.
    为了能够运行这个程序,内部类的对象总有一个隐式引用,它指向了创建它的外部类对象.
    这个引用在内部类的定义中是不可见的,然而,为了说明这个概念,我们将外围类对象的引用称为outer,于是actionPerformed方法等价于下列形式:
public void actionPerformed(ActionEvent event)
{
    Date now = new Date();
    System.out.println("At the tone, the time is " + now);
    if (outer.beep)
        Toolkit.getDefaultToolkit().beep();
}
    外围类的引用在构造器中设置,编译器修改了所有的内部类的构造器,添加一个外围类引用的参数,因为TimePrinter类没有定义构造器,所以编译器为这个类生成了一个默认的构造器,其代码如下所示:
public TimePrinter(TalkingClock clock)
{
    outer = clock;
}
    注意,outer不是Java的关键字,我们只是用它说明内部类中的机制.
    当在start方法中创建了TimePrinter对象后,编译器就会将 this 引用传递给当前的语音时钟的构造器:
ActionListener listener = new TimePrinter(this);    
    注释:TimePrinter类声明为私有的,这样一来,只有TalkingClock的方法才能构造TimePrinter对象, 只有内部类可以是私有类,而常规类只可以具有包可见性,或公有可见性.
    innerClass/InnerClassTest.java如下所示:
package innerClass;

import javax.swing.*;

/**
 * This program demonstrates the use of inner classes.
 */
public class InnerClassTest
{
    public static void main(String[] args)
    {
        TalkingClock clock = new TalkingClock(1000, true);
        clock.start();

        // keep program running until user selects "OK"
        JOptionPane.showMessageDialog(null, "Quit program?");
        System.exit(0);
    }
}
    innerClass/TalkingClock.java如下所示:
package innerClass;

import java.awt.*;
import java.awt.event.*;
import java.util.*;
import javax.swing.*;
import javax.swing.Timer;

public class TalkingClock
{
    private int interval;
    private boolean beep;

    /**
     * Constructs a talking clock
     * @param interval the interval between messages(in milliseconds)
     * @param beep true if the clock should beep
     */
    public TalkingClock(int interval, boolean beep)
    {
        this.interval = interval;
        this.beep = beep;
    }
    
    public void start()
    {
        ActionListener listener = new TimePrinter();
        Timer t = new Timer(interval, listener);
        t.start();
    }

    public class TimePrinter implements ActionListener
    {
        public void actionPerformed(ActionEvent event)
        {
            Date now = new Date();
            System.out.println("At the tone, the time is " + now);
            if (beep)
                Toolkit.getDefaultToolkit().beep();
        }
    }
}
    运行结果如下所示:

6.4.2   内部类的特殊语法规则

    在上一节中,已经讲述了内部类有一个外围类的引用outer,事实上,使用外围类引用的正规语法还要复杂一些.表达式
OuterClass.this
    表示外围类引用.例如,可以像下面这样编写TimePrinter内部类的actionPerformed方法:
public void actionPerformed(ActionEvent event)
{
    ...
    if (TalkingClock.this.beep)
        Toolkit.getDefaultToolkit().beep();
}
    反过来,可以采用下列语法格式更加明确地编写内部对象的构造器:
outerObject.new InnerClass(construction parameters);
    例如,
ActionListener listener = this.new TimePrinter();
    在这里,最新构造的TimePrinter对象的外围类应用被设置为内部类对象的方法中的 this 引用.这是一种最常见的情况,通常,this 限定词是多余的.不过可以通过显式地命名将外围类引用设置为其他的对象.例如,如果TimePrinter是一个公有内部类,对于任意的语音时钟都可以构造一个TimePrinter:
TalkingClock jabberer = new TalkingClock(1000, true);
TalkingClock.TimePrinter listener = jabberer.new TimePrinter();
    需要注意,在外围类的作用域之外,可以这样应用内部类:
OuterClass.InnerClass

6.4.3   内部类是否有用,必要和安全

    内部类是一种编译器现象,与虚拟机无关. 编译器将会把内部类翻译成用$分隔外部类名与内部类名的常规类文件,而虚拟机则对此一无所知.
    例如,在TalkingClock类内部的TimePrinter类将翻译成类文件TalkingClock$TimePrinter.class .
    如下所示(在TalkingClock与TimePrinter之间有一个$符号, 说明TimePrinter是TalkingClock的内部类):

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值