【Java】 内部类

【Java】内部类

可以将一个类的定义放在另一个类的定义内部,这就是内部类。
使用内部类的的原因主要有三点:

  • 内部类方法可以访问该类定义所在的作用域中的数据,包括私有的数据。
  • 内部类可以对同一个包中的其他类隐藏起来。
  • 当想要定义一个回调函数且不想编写大量代码时,使用匿名(anonymous)内部类比较便捷

使用内部类访问对象状态

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

    public TalkingClock(int interval, boolean beep) {...}
    public void start() {...}
    
    public class TimePrinter implements ActionListener {
        Date now = new Date();
        System.out.println("At the tone, this time is" + now);
        if (beep) Toolkit.getDefaultToolkit().beep();
    }
}

内部类既可以访问自身的数据域,也可以访问创建它的外围类对象的数据域。

外围类的引用在构造器中设置。编译器修改了所有的内部类的构造器,添加一个外围类引用的参数。因为TimePrinter类没有定义构造器,所以编译器为这个类生成了一个默认的构造器,其代码如下:

public TimePrinter(TalkingClock clock) {
    outer = clock;
}

请再注意一下,outer不是Java的关键字。我们只是用它说明内部类中的机制。
当在start方法中创建了TimePrinter对象后,编辑器就会将this引用传递给当前的语音始终的构造器。

ActionListener listener = new TimePinter(this); //parameter automatically added

内部类的特殊语法规则

使用外围类引用的正规语法如下。表达式:
OuterClass.this
表示外围类引用。例如,可以像下面这样编写TimePrinter内部类的actionPerformed方法:

public void actionPerformed(ActionEvent event) {
    if (TalkingClock.this.beep) Toolkit.getDefaultToolkit().beep();
}

反过来,可以采用下列语法格式更加明确地编写内部对象的构造器:

outerClass.new InnerClass(construction parameters)
例如
Actionlistner listener = this.new TimePrinter();

在这里,最新构造的TimePrinter对象的外围类引用被设置为创建内部类对象的方法中的this引用。这是一种最常见的情况。通常,this限定词是多余的。不过,可以通过显式地命名将外围类引用设置为其他的对象。例如,如果TimePrinter是一个共有内部类,对于任意的语音时钟都可以构造一个TimePrinter:

TalkingClock jabberer = new TalkingClock(1000, true);
Talking.TimePrinter listener = jabber.new TimePrinter();

需要注意,在外围类的作用域之外,可以这样引用内部类:
OuterClass.InnerClass

要想直接创建内部类的对象,你不能按照你想象的方式,去引用外部类的名字,而必须使用外部类的对象来创建该内部类对象。在拥有外部类对象之前是不可能创建内部类对象的。这是因为内部类对象会暗暗地连接到创建它的外部类对象上。


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

  • 内部类是一种编译器现象,与虚拟机无关。编译器会把内部类翻译成用$(美元符号)分隔外部类与内部类名的常规类文件,而虚拟机一无所知。例如,在TalkingClock类内部的TimePrinter类将被翻译成TalkingClock$TimePrineter.class。
  • 如果一个类是匿名内部类,那么clas文件名称是OuterClass$(1,2,3).class
  • 编译器为了引用外部类,生成了一个附加的实例域this$0(名字this$0是由编译器合成的,在自己编写的代码中不能够引用)。

局部内部类

假设TimePrinter这个类名字只在start方法中创建这个类型的对象时使用了一次,那么可以像下面这样使用:

public void start() {
    class TimePrinter implements ActionListner {
        public void actionPerformed(ActionEvent event) {
            Date now = new Date();
            System.out.println("At the tone, the time is " + now);
            if (beep) Toolkit.getDefaultToolkit().beep();
        }
    }
    ActionListener listener = new TimePrinter();
    Timer t = new Timer(interval, listner);
    t.start();
}

局部类不能用public或private访问说明符进行声明。它的作用域被限定在声明这个局部类的块中。
局部类有一个优势,即对外部世界可以完全地隐藏起来。即使TaklingClock类中的其他代码也不能访问。除start方法之外,没有任何方法知道TimePrinter类的存在。


由外部方法访问final变量

与其他内部类比较,局部类还有一个优点。它们不仅能够访问包含它们的外部类,还可以访问局部变量。不过,那些局部变量必须被声明为final。如:

public void start(int interval, final bolean beep) {
    class TimePrinter implements ActionListner {
        public void actionPerformed(ActionEvent event) {
            Date now = new Date();
            System.out.println("At the tone, the time is " + now());
            if (beep) Toolkit.getDefaultToolkit().beep();
        }
     }
}

编译器实现内部类访问局部变量的方式是这样的:在内部类中为每一个要访问的局部变量设置数据域,然后在构造函数中将这些数据域初始化为要访问的局部变量值。

匿名内部类

将局部内部类的使用再深入一步。假如只创建这个类的一个对象,就不必命名了。这种类被称为匿名内部类(annoymous inner class)

public void start(int interval, final boolean beep) {
    ActionListener listener = new 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();
        }
    }
   Timer t = new Timer(interval, listner);
   t.start();
}

它的含义是:创建一个实现ActionListner接口的类的新对象,需要实现的方法actionPerformed定义在括号{}内。
通常的语法格式为:

    new SuperType(construction parameters) {
        inner class methods and data
    }

其中,SuperType可以是ActionListner这样的接口,于是内部类就要实现这个接口。SuperType也可以是一个类,于是内部类就要扩展它。
如果你的基类需要一个有参数的构造器,应该怎么办:

public classs Parcel8 {
    public Wrapping wrapping(int x) {
        //Base constructor call
        return new Wrapping(x) {
            public int value() {
                return super.value() * 47;
            }
        };
    public static void main(String[] args) {
        Parcel8 p = new Parcel8();
        Wrapping w = p.wrapping(10);
    }
}

只需要简单地传递合适的参数给基类的构造器即可,这里是将x传进new Wrapping(x)。尽管Wrapping 只是一个具有具体实现的普通类,但它还是被其导出类当做公共“接口”来使用

    public class Wrapping {
        private int i;
        public Wrapping (int x) { i = x;}
        public int value() { return i; }
    }

你会注意到, Wrapping拥有一个要求传递一个参数的构造器,这使得事情变的更有趣了。
在匿名类中定义定义字段时,还能够对其执行初始化操作:

    public class Parcel9 {
        public Destination destination(final String dest) {
            return new Destination() {
                private String label = dest;
                public String readLabel() { return label; }
            };
        }
    public static void main(String[] args) {
        Parcel9 p = new Parcel9();
        Destination d = p.destination(“Tesmania”);
    }
}

如果定义一个匿名内部类,并且希望它使用一个在其外部定义的对象,name编译器会要求其参数引用是final的,就像你在destinaion()的参数中看到的那样。如果你忘记了,将会得到一个错误信息。
如果只是简单地给一个字段赋值,那么此例中的方法是很好的。但是,如果想做一些类似构造器的行为。在匿名类中不可能有命名的构造器,但通过实例初始化,就能够达到为匿名内部类创建一个构造器的效果,就像这样:

abstract class Base {
    public Base(int i) {
        print("Base conctructor, i=" + 1);
    }
}

public class AnonymousConstructor {
    public class Base getBase(int i){
        return new Base(i){
            print("Inside instance initializer");
            
            public void f() {
                print("In anonymous f()");
            }
        }
    }
    
    public static void main(String[] args) {
        Base base = getBase(47);
        base.f();
    }
}/*
Base constructor, i = 47;
Inside instance initializer
In  anonymous f()

在此例中,不要求变量i一定是final的。以为i被传递给匿名内部类的积累的构造器,它并不会在你你们内部类被直接使用。
下例是带实例化的“parcel”形式。注意destination()参数必须是final的,因为它们是在匿名内部类使用的:

public class Parcel10 {
    public Destination destination(final String dest, final float price) {
        return new Destination(){
            private int cost;
            //Instance initialization for each object
            {
                cost = Math.round(price);
                if (cost > 100) {
                    System.out.println("Over budget");
                }
            }
            private String label = dest;
    
            public String readLabel() {
                return label;
            }
        };
        
        
    }
    
    public static void main(String[] args) {
        Parcel10 p = new Parcel10();
        Destination d = p.destination("Tasmania", 101.395.F);
    }
}
//
Over budget;
匿名内部类的重点

1.使用匿名内部类时,必须继承或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或者实现一个接口
2. 匿名内部类是不能定义构造函数的
3. 匿名内部类是不能存在任何的静态成员变量和静态方法
4. 匿名内部类为局部内部类,所有局部内部类的所有限制对匿名内部类生效
5. 匿名内部类不能使抽象的,它必须要实现继承的类或者实现接口的所有抽象方法

嵌套类

  • 如果不需要内部类对象与其外围类对象之间有联系,那么可以将内部类声明为staic。这通常称为嵌套类

  • 普通的内部类对象隐式地保存了一个引用,指向创建它的外围类对象。然而,当内部类是static的时,就不是这样了,嵌套类意味着:
    1)要创建嵌套类的对象,不需要其外围类的对象。
    2)不能从嵌套类的对象中访问非静态的外围类对象。

  • 嵌套类和普通的内部类还有一个区别。普通内部类的字段与方法,只能放在类的外部层次,所以普通的内部类不能有static数据和static字段,也不能包含嵌套类。但是嵌套类可以包含这些东西。

闭包和回调

  • 闭包是一个可调用的对象,它记录了一些信息,这些信息来自于创建它的作用域。通过这个定义,可以看出内部类是面向对象的闭包,因为它不仅包含外围类的对象的信息,还自动拥有一个指向此外围类对象的引用,在此作用域内,内部类有权操作所有的成员,包括private成员。

内部类的重点

  1. 非静态内部类可以访问外部类的数据域,包括私有的
  2. 局部内部类和匿名内部类可以访问方法中的参数,不过参数必须为final
  3. 匿名内部类,必须要继承一个类或者实现一个接口,但是两者不可兼得,同时也只能继承一个类或实现一个接口。
  4. 匿名内部类不能定义构造函数
  5. 匿名内部类中不能存在任何的静态成员变量和静态方法。
posted @ 2017-01-07 16:24  vincently  阅读(296)  评论(0编辑  收藏  举报