优先考虑静态成员类

嵌套类(nested class)是指被定义在另一个类的内部的类。嵌套存在的目的应该只是为它的外围类提供服务。如果嵌套类将来可能会用于其他的某个环境中,它就应该是顶层类。嵌套类有四种:静态成员类,非静态成员类,匿名类和局部类,除了第一种外,其他三种都被称为内部类。本条目将告诉你什么时候应该使用哪种嵌套类,以及这样做的原因。

静态成员类是最简单的一种嵌套类。最好把它看作是普通的类。只是碰巧被声明在另一个类的内部而已,它可以访问外围类的所有成员,包括那些申明为私有的成员。静态成员类是外围类的一个静态成员,与其他的静态成员一样,也遵守同样的可访问性规则。如果它被申明为私有的,它就只能在外围类的内部才可以被访问,等等。

静态成员类的一种常见用法是作为共有的辅助类,仅当与它的外部类一起使用时才有意义。

public class Outer {
    private String name;
    private int age;

    public static class Builder {
        private String name;
        private int age;

        public Builder(int age) {
            this.age = age;
        }

        public Builder withName(String name) {
            this.name = name;
            return this;
        }

        public Builder withAge(int age) {
            this.age = age;
            return this;
        }

        public Outer build() {
            return new Outer(this);
        }
    }

    private Outer(Builder b) {
        this.age = b.age;
        this.name = b.name;
    }
}

从语法上来讲,静态成员类和非静态成员类之间唯一的区别是,静态成员类的声明中包含修饰符static,尽管他们的语法非常相似,但是这两种嵌套类有很大的不同。非静态成员类的每个实例都隐含着与外围实例上的方法,或者利用修饰过的this构造获得外围实例的引用。如果嵌套类的实例可以在它外围类的实例之外独立存在,这个嵌套类就必须是静态成员类,在没有外围实例的情况下,要想创建非静态成员的实例是不可能的。

当非静态成员类的实例被创建的时候,它和外围实例之间的关联关系也随之被建立起来,而且,这种关联关系以后不能被修改。通常情况下,当外围类的某个实例方法的内部调用非静态成员类的构造器时,这种关联关系被自动建立起来,这种关联关系需要消耗非静态成员类实例的空间,并且增加了构造的时间开销。如果声明成员类不要求访问外围实例,就要始终把static修饰符放在它的声明中,使它成为静态成员类,而不是非静态成员类。如果省略了static修饰符,则每个实例都将包含一个额外的指向外围对象的引用。保存这份引用要消耗时间和空间,并且会导致外围实例在符合垃圾回收时却任然保留

非静态成员类的常见用法是实现各种集合的迭代器(iterator),私有静态成员类的常见用法是用来代表外围类的对象的组件。例如,考虑一个Map实例,它把key和value关联起来,许多Map实现的内部都有一个Entry对象
对应于Map中的咩歌键值对。虽然每个entry都与一个map关联,但是entry上的方法(getKey,getValue等)并不需要访问改Map,因此使用非静态成员来表示就很浪费,

匿名类

匿名类不同于Java程序设计语言中的其他任何语法单元。正如你所想象的,匿名类没有名字。它不是外围类的一个成员。它并不与其他的成员一起被声明,而是在使用的同时被声明和实例化。匿名类可以出现在代码中任何允许存在表达式的地方。当且仅当匿名类出现在非静态的环境中时,它才有外围实例。但是即使它们出现在静态的环境中,也不可能拥有任何静态成员。
匿名类的适用性受到诸多的限制。

  • 除了在它们被声明的时候之外,是无法将它们实例化的,你不能执行instanceof测试,或者做任何需要命名类的其他事情。
  • 你无法声明一个匿名类来实现多个接口,或者扩展一个类,并同时扩展类和实现接口。
  • 匿名类的客户端无法调用任何成员,除了从它的超类型中继承得到之外。
  • 由于匿名类出现在表达式当中,它们必须保持简短——大约10行或者更少些——否则会影响程序的可读性。

匿名类常见用法

  • 第一种常见用法就是动态的创建函数对象(function object,见21条)。例如,第21条中-Arrays.sort方法调用,利用匿名的Comparator实例,根据一组字符串的长度对它们进行排序。
  • 第二种常见的用法是创建过程对象(process object),比如Runnable、Thread或者TimerTask实例。
  • 第三种常见的用法是在静态工厂内部(参见第18条中部分的intArrayAsList方法)。
static List<Integer> intArrayAsList(final int[] a) {  

    if (a == null)  

        throw new NullPointerException();  

    return new AbstractList<Integer>() {  

        public Integer get(int i) {  

            return a[i];  

        }  

        @Override  

        public Integer set(int i, Integer val) {  

            int oldVal = a[i];  

            a[i] = val;  

            return oldVal;  

        }  

        public int size() {  

            return a.length;  

        }  

    };  

}

局部类

局部类是四种嵌套类中用的最少的类。在任何“可以声明局部变量”的地方,都可以声明局部类,并且局部类也遵守同样的作用域规则。局部类与其他三种嵌套类中的每一种都有一些共同的属性。与成员类一样,局部类有名字,可以被重复使用。与匿名类一样,只有当局部类实在非静态环境中定义的时候,才有外围实例,它们也不能包含静态成员。与匿名类一样,它们必须简短以便不会影响到可读性。
示例:

public class outerTolocal {

    public String string;

    public int localInt;

    public void OtoLocal() {}

    public void localMthod(final int m, int n) {
        class local {
            //此类为局部类
            //局部类不需要加public 修饰符,因为这方法执行完 这类就消失了
            int methodInt = m;
            /**
             * 局部类的变量如果要等于外部类的方法的变量,
             * 此时外部类的方法变量必须用final 修饰符
             * 如:
             */
            final int m;

            void localInner() {
                System.out.println("local method");
            }
        }
        new local().localInner();//在另外的一个类的中不可以创建局部内部类的实例,只能在局部内部类中来创建。
    }
}
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,133评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,682评论 3 390
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,784评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,508评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,603评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,607评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,604评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,359评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,805评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,121评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,280评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,959评论 5 339
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,588评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,206评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,442评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,193评论 2 367
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,144评论 2 352

推荐阅读更多精彩内容

  • 概念 嵌套类(nested class) 指被定义在另一个类的内部的类。嵌套类存在的目的应该只是为他的外围类(en...
    NekoJiang阅读 1,261评论 0 2
  • 抽象类 在继承的层次结构中,每个新子类都使类变得越来越明确具体。如果从一个子类追溯到父类,类就会变得更通用和抽象。...
    Steven1997阅读 1,374评论 0 5
  • 目录 第二章 创建和销毁对象 1 考虑用静态工厂方法替代构造器 对于代码来说, 清晰和简洁是最重要的. 代码应该被...
    高广超阅读 1,447评论 0 12
  • 01 没有谁想成为一座孤岛,内向者也一样。 内向者普遍被认为是一个不活跃,不合群的群体,与生俱来的性格使这群人很难...
    青橙梓阅读 1,095评论 3 2
  • 今天让我尝到了最大的挫败感,让我终于知道了不好好学习的后果。唉,悔之晚矣。现在的我只想安下心来,好好的把东西学好,...
    我的回忆录阅读 140评论 0 0