深入理解Java内部类

     内部类就是定义在一个类中的另外一个类,是一种从属关系。在没有实际了解内部类之前,我始终困惑,为什么要在一个类中定义另外一个类,这不是增加代码结构复杂度么?现在才大致能知道这种设计的优势是大于其劣势的。比如,我们可以通过内部类解决类的单继承问题,外部类不能再继承的类可以交给内部类继承。我们可以通过定义内部类来实现一个类私属于一个类,实现更好的封装性。具体的我们接下来介绍,本文主要通过介绍内部类的四种不同类型的定义,实例的创建,内部实现原理以及使用场景几种不同角度来学习内部类。

  • 静态内部类
  • 成员内部类
  • 方法内部类
  • 匿名内部类

一、静态内部类
     静态内部类的定义和普通的静态变量或者静态方法的定义方法是一样的,使用static关键字,只不过这次static是修饰在class上的,一般而言,只有静态内部类才允许使用static关键字修饰,普通类的定义是不能用static关键字修饰的,这一点需要注意一下。下面定义一个静态内部类:

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

    public static class In{
        private int age;
        public void sayHello(){
            
            System.out.println("my name is : "+name);
            //--编译报错---
            //System.out.println("my age is :"+ age);
        }
    }
}

在上述代码中,In这个类就是一个静态内部类。我们说内部类是可以访问外部类的私有字段和私有方法的,对于静态内部类,它遵循一致的原则,只能访问外部类的静态成员。上述代码中,外部类的非静态私有字段age在静态内部类中使不允许访问的,而静态字段name则是可访问的。下面我们看,如何创建一个静态内部类的实例对象。

public static void main(String [] args){
    Out.In innerClass = new Out.In();
    innerClass.sayHello();
}

静态内部类的实例对象创建还是比较简洁的,不同于成员内部类,它不需要关联外部类实例(具体的下文介绍),下面我们再看一段代码:

public class Out {
    private static String name;

    public static class In{
        public void sayHello(){
            System.out.println(name);
            showName();
        }
    }

    private static void showName(){
        System.out.println(name);
    }
}

上述代码在内部类中两次访问了外部类的静态成员,第一次访问了静态字段name,第二次访问的静态方法showName。在我们反编译这个类之前,首先需要知道的是,所谓的内部类的概念只是出现在编译阶段,对于jvm层是没有内部类这个概念的。也就是说,编译器会将一个类编译成一个源文件,对于内部类也是一样,它会从它的外部类中抽离出来,增加一些与外部类的联系,然后被编译成一个单独的源文件。下面我们先编译运行之后,利用Dj反编译class文件看看编译器都做了些什么事情。

//这是我们的Out外部类
public class Out
{
    //省去了一些不重要的部分
    private static void showName()
    {
        System.out.println(name);
    }

    private static String name;
    
    static String access$000(){return name;}
    static void access$100(){showName();}

}
//这是我们的内部类
public static class Out$In
{

    public void sayHello()
    {
        System.out.println(Out.access$000());
        Out.access$100();
    }

    public Out$In()
    {
    }
}

相信大家也已经看出来这两者之间的某种联系,编译器将Out这个类编译成两个独立的class源文件。对于Out中所有的私有成员(也就是内部类分离出去之后不能访问的成员),增设了可供调用的access$xxx方法,从而实现内部类与外部类之间的联系。这就是他们的本质。

至于使用场景,一般来说,对于和外部类联系紧密但是并不依赖于外部类实例的情况下,可以考虑定义成静态内部类。下面我们看稍显复杂的成员内部类。

二、成员内部类
     我们说了,四种不同类型的内部类都各自有各自的使用场景,静态内部类适合于那种和外部类关系密切但是并不依赖外部类实例的情况。但是对于需要和外部类实例相关联的情况下,可以选择将内部类定义成成员内部类。以下代码定义了一个简单的成员内部类:

public class Out {
    private String name;

    public void showName(){
        System.out.println("my name is : "+name);
    }

    public class In{
        public void sayHello(){
            System.out.println(name);
            Out.this.showName();
        }
    }
}

以上定义了一个简单的内部类In,我们的成员内部类可以直接访问外部类的成员字段和成员方法,因为它是关联着一个外部类实例的。下面我们看看在外部是如何创建该内部类实例的。

public static void main(String [] args){
    Out out = new Out();
    Out.In in = out.new In();
    in.sayHello();
}

因为成员内部类是关联着一个具体的外部类实例的,所以它的实例创建必然是由外部类实例来创建的。对于实例的创建,我们只需要记住即可,成员内部类的实例创建需要关联外部类实例对象,静态内部类实例创建相对简单。下面我们主要看看在编译阶段编译器是如何保持内部类对外部类成员信息可访问的。

//反编译的Out外部类源码
public class Out
{
    //省略部分非核心代码
    public void showName()
    {
        System.out.println((new StringBuilder()).append("my name is : ").append(name).toString());
    }

    private String name;
    
    static String access$000(Out o){return o.name;}
}
//反编译的内部类In源码
public class Out$In
{
    public void sayHello()
    {
        System.out.println(Out.access$000(Out.this));
        showName();
    }

    final Out this$0;

    public Out$In()
    {
        this.this$0 = Out.this;
        super();
    }
}

由上述代码其实我们可以知道,当我们利用外部类实例创建内部类实例的时候,会将外部类实例作为初始资源传入内部类构造过程。这样我们就可以通过该实例访问外部类所有的成员信息,包括私有成员。(显式增加了暴露方法)

至于使用场景,对于那种要高度依赖外部类实例的情况下,定义一个成员内部类则会显的更加明智。

三、方法内部类
     方法内部类,顾名思义,定义在一个方法内部的类。方法内部类相对而言要复杂一些,下面定义一个方法内部类:

public class Out {
    private String name;

    public void sayHello(){
        class In{
            public void showName(){
                System.out.println("my name is : "+name);
            }
        }

        In in = new In();
        in.showName();
    }
}

我们定义了一个类,在该类中又定义了一个方法sayHello,然而在该方法中我们定义了一个内部类,类In就是一个方法内部类。我们的方法内部类的生命周期不超过包含它的方法的生命周期,也就是说,方法内部类只能在方法中使用。所以在声明的时候,任何的访问修饰符都是没有意义的,于是Java干脆不允许使用任何的访问修饰符修饰方法内部类。其中还需要注意一点的是,定义和使用时两回事,别看那一大串定义类的代码,你实际想要使用该类,就必须new对象,而对于方法内部类而言,只能在方法内部new对象。这就是方法内部类的简单介绍,下面我们看看其实现原理。

有关方法内部类的实现原理其实是和成员内部类差不太多的,也是在内部类初始化的时候为其传入一个外部类实例,区别在哪呢?就在于方法内部类是定义在具体方法的内部的,所以该类除了可以通过传入的外部实例访问外部类中的字段和方法,对于包含它的方法中被传入的参数也会随着外部类实例一起初始化给内部类。

毋庸置疑的是,方法内部类的封装性比之前介绍的两种都要完善。所以一般只有在需要高度封装的时候才会将类定义成方法内部类。

四、匿名内部类
     可能内部类的所有分类中,匿名内部类的名号是最大的,也是我们最常用到的,多见于函数式编程,lambda表达式等。下面我们重点看看这个匿名内部类。

匿名内部类就是没有名字的内部类,在定义完成同时,实例也创建好了,常常和new关键字紧密结合。当然,它也不局限于类,也可以是接口
,可以出现在任何位置。下面我们定义一个匿名内部类:

//首先定义一个普通类
public class Out {
    private String name;

    private void sayHello(){
        System.out.println("my name is :" + name);
    }
}
//定义和使用一个匿名内部类
public static void main(String [] args){
    Out out = new Out(){
        @Override
        public void sayHello(){
            System.out.println("my name is cyy");
        }
        public void showName(){
            System.out.println("hello single");
        }
    };
    out.sayHello();
}

从上述代码中可以很显然的让我们看出来,我们的匿名内部类必定是要依托一个父类的,因为它是没有名字的,无法用一个具体的类型来表示。所以匿名内部类往往都是通过继承一个父类,重写或者重新声明一些成员来实现一个匿名内部类的定义。实际上还是利用了里式转换原理。

从中我们也可以看到,一个匿名内部类定义的完成就意味着该内部类实例创建的完成。下面我们看看其实现原理:

//反编译出来的匿名内部类
static class Test$1 extends Out
{
    Out out;
    public void sayHello()
    {
        System.out.println("my name is cyy");
    }

    Test$1(Out o)
    {
        this.out = o;
    }
}

其实在看了上述三种内部类的原理之后,反而觉得匿名内部类的实现较为简单了。主要思路还是将内部类抽离出来,通过初始化传入外部类的实例以达到对外部类所有成员的访问。只是在匿名内部类中,被依托的父类不是他的外部类。匿名内部类的主要特点在于,没有名字,对象只能被使用一次,可以出现在任意位置。所以它的使用场景也是呼之欲出,对于一些对代码简洁度有所要求的情况下,可首选匿名内部类。

以上完成了对四种内部类的简单介绍,对于他们各自实现的原理也都已经介绍过了。其实大致相同,由于jvm对每个类都要求一个单独的源码文件,所以编译阶段就完成了分离的操作,但是在分离的过程中又要保持内部类和外部类之间的这种联系,于是编译器添加了一些接口保持这种信息共享的结构。使用内部类可以大大增加程序的封装性,使得代码整体简洁度较高。

本文主要来自于多篇文章的学习,总结不到之处,望指出!

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

推荐阅读更多精彩内容

  • 前言:对于 JAVA 基础的内容,其实网上资料挺多的,本打算只写 Android 相关技术文章,但对于内部类这个基...
    theonlin阅读 1,347评论 0 12
  • Java 内部类 分四种:成员内部类、局部内部类、静态内部类和匿名内部类。 1、成员内部类: 即作为外部类的一个成...
    ikaroskun阅读 1,213评论 0 13
  • 前言:对于 JAVA 基础的内容,其实网上资料挺多的,本打算只写 Android 相关技术文章,但对于内部类这个基...
    theonlin阅读 5,155评论 7 13
  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,537评论 18 399
  • 《你的降落伞是什么颜色》这本书号称是全球最畅销的求职领域跳槽指南,被誉为“求职圣经”,它除了提供求职、跳槽技巧,也...
    阿同学阅读 265评论 0 3