zoukankan      html  css  js  c++  java
  • 【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. 匿名内部类中不能存在任何的静态成员变量和静态方法。
  • 相关阅读:
    站点防攻击之禁止国外ip
    docker 安装rabbitMQ
    【转】浅析支付系统的整体架构
    【转】MQ详解及四大MQ比较
    SQL server 中的dbo、guest
    网站加载速度的优化(1):优化前端页面
    【转】Apache ab压力测试
    Asp.Net Core 部署发布后EFCore数据库断连和Google浏览器无法存储Cookie
    C# 使用nuget.exe发布类库及更新类库
    .net core 读取Excal文件数据及注意事项
  • 原文地址:https://www.cnblogs.com/vincently/p/6259673.html
Copyright © 2011-2022 走看看