Java & Groovy & Scala & Kotlin - 17.内部类

Overview

所谓的内部类即定义在类内部的类,而包含这个内部类的类则被称作外部类。通常来说内部类可以访问外部类的私有成员,作为外部类的内部扩展而存在。

Java 篇

静态内部类

静态内部类即以 static 关键字声明的内部类。静态内部类不属于外部类的成员,使用上与普通的外部类没有什么区别。

定义一个静态内部类

class Outter {
    static class StaticInner {
    }
}

创建静态内部类的实例

Outter.StaticInner staticInner = new Outter.StaticInner();

匿名内部类

匿名内部类即没有指明名字的内部类,通常用于监听器和线程的创建。

例:

Thread thread = new Thread(new Runnable() {
    @Override
    public void run() {

    }
});

以上 new Runnable() 即定义了一个没有名字的实现了 Runnalbe 接口的内部类并立即创建该类的实例作为 Thread 的构造方法的参数。

非静态内部类

非静态内部类属于外部类,可以被看做为外部类的一个成员,和外部类定义的普通方法和成员变量没有什么区别,所以在该内部类中可以访问外部类的所有成员。也正是这个原因,创建内部类的实例时必须先有外部类的实例。

定义一个非静态内部类

如果内部类和外部类拥有同名成员变量,如果直接使用变量名访问,遵循就近原则,方法的只可能是内部类自身的成员。但是可以通过 外部类.this.变量名 的方式来明确指明需要调用的是外部类成员而不是内部类的成员。

例:

class Outter {

    private String name;

    public Outter(String name) {
        this.name = name;
    }

    public class Inner {
        private String name;

        public Inner(String name) {
            this.name = name;
        }

        public String desc() {
            return Outter.this.name + "-" + name;
        }
    }

    public void foo(Inner bar) {
        System.out.println(bar.desc());
    }
}

使用该内部类

Outter outter1 = new Outter("Outter1");
Outter.Inner inner1 = outter1.new Inner("Inner1");

Outter outter2 = new Outter("Outter2");
Outter.Inner inner2 = outter2.new Inner("Inner2");

System.out.println("outter1 is " + outter1.getClass()); //  _innerclass.Outter
System.out.println("outter2 is " + outter2.getClass()); //  _innerclass.Outter
System.out.println("inner1 is " + inner1.getClass());   //  _innerclass.Outter$Inner
System.out.println("inner2 is " + inner2.getClass());   //  _innerclass.Outter$Inner

outter1.foo(inner1);    //  Outter1-Inner1
outter1.foo(inner2);    //  Outter2-Inner2

以上可以看到 inner1inner2 两个通过不同外部类实例创建的内部类对象并没有什么区别,可见内部类是属于外部类的。

Groovy 篇

静态内部类

使用同 Java

定义一个静态内部类

class Outter {
    static class StaticInner {
    }
}

创建静态内部类的实例

def staticInner = new Outter.StaticInner()

匿名内部类

匿名内部类使用也同 Java

例:

def thread = new Thread(new Runnable() {
    @Override
    void run() {

    }
})

非静态内部类

非静态内部类使用也同 Java。但是,在 Groovy 中创建内部类的实例时语法与 Java 不同,使用的是 new 外部类.内部类(外部类的实例 [, 内部类的构造方法的参数]),且此时指明内部类的构造方法参数时不能使用带名参数。

定义一个非静态内部类

class Outter {

    def name

    static class StaticInner {
    }

    class Inner {
        def name
        Inner(def name) {
            this.name = name
        }
        def desc = "${Outter.this.name}-${name}"
    }

    def foo(Inner bar) {
        println(bar.desc)
    }
}

使用该内部类

def outter1 = new Outter(name: "Outter1")
//  Wrong!! 不能使用带名参数
//        def inner1 = new Outter.Inner(outter1, name: "Inner1")
def inner1 = new Outter.Inner(outter1, "Inner1")

def outter2 = new Outter(name: "Outter2")
def inner2 = new Outter.Inner(outter2, "Inner2")

println("outter1 is ${outter1.getClass()}") //  _innerclass.Outter
println("outter2 is ${outter2.getClass()}") //  _innerclass.Outter
println("inner1 is ${inner1.getClass()}")   //  _innerclass.Outter$Inner
println("inner2 is ${inner2.getClass()}")   //  _innerclass.Outter$Inner

outter1.foo(inner1) //  Outter1-Inner1
outter1.foo(inner2) //  Outter2-Inner2

由以上例子可见在 Groovy 中内部类也是属于外部类的。

Scala 篇

匿名内部类

匿名内部类使用也同 Java

val thread = new Thread(new Runnable {
  def run(): Unit = {

  }
})

非静态内部类

Scala 中非静态内部类与 Java 中有很大不同。Scala 中非静态内部类是属于外部类的实例,而不是外部类自身。

定义一个非静态内部类

Scala 中内部类也可以通过 外部类.this.成员名 来访问外部类的成员,或者也可以通过像如下的 outter => 定义一个 外部类.this 的别名来访问。

class Outter(val name: String) {
  outter =>

  class Inner(val name: String) {
    def desc = s"${Outter.this.name}-$name"

    def desc2 = s"${outter.name}-$name"
  }

  def foo(bar: Inner): Unit = {
    println(bar.desc)
  }
}

使用该内部类

val outter1 = new Outter("Outter1")
val inner1 = new outter1.Inner("Inner1")

val outter2 = new Outter("Outter2")
val inner2 = new outter2.Inner("Inner2")

println(s"outter1 is ${outter1.getClass}") //  _innerclass.Outter
println(s"outter2 is ${outter2.getClass}") //  _innerclass.Outter
println(s"inner1 is ${inner1.getClass}")  //  _innerclass.Outter$Inner
println(s"inner2 is ${inner2.getClass}")  //  _innerclass.Outter$Inner

以上打印时可以看到 inner1inner2 的类型看起来是一样的。但是调用以下方法后会发现报 type mismatch 错误。

outter1.foo(inner1)
//  Wrong!! type mismatch
//  outter1.foo(inner2)  

实际上这是由于 Scala 中存在着一种被称作 "路径依赖类型" 的概念,即 "A.this.B" 随着 A 是不同的实例而不同。所以以上定义的 foo() 方法只能接受特定路径的 Inner 类的实例。

如果希望 foo() 方法接收所有 Outter 实例路径下的 Inner 类型,可以使用类型投影。类型投影使用符号 # 来定义。

例:

foo() 方法换成以下形式

def foo(bar: Outter#Inner): Unit = {
  println(bar.desc2)
}

再调用以下方法就不会报错了

outter1.foo(inner1)
outter1.foo(inner2)

Kotlin 篇

匿名内部类

匿名内部类使用也同 Java

val thread = Thread(Runnable {

})

非静态内部类

Kotlin 中非静态内部类与 Java 相似。

定义一个非静态内部类

Kotlin 中内部类使用语法 this@外部类.成员名 来调用外部成员。

class Outter(val name: String) {

    inner class Inner(val name: String) {
        fun desc() = "${this@Outter.name}-${name}"
    }

    fun foo(bar: Inner) {
        println(bar.desc())
    }
}

使用该内部类

val outter1 = Outter("Outter1")
val inner1 = outter1.Inner("Inner1")

val outter2 = Outter("Outter2")
val inner2 = outter1.Inner("Inner2")

println("outter1 is ${outter1.javaClass}")  //  _innerclass.Outter
println("outter2 is ${outter2.javaClass}")  //  _innerclass.Outter
println("inner1 is ${inner1.javaClass}")    //  _innerclass.Outter$Inner
println("inner2 is ${inner2.javaClass}")    //  _innerclass.Outter$Inner

outter1.foo(inner1) //  Outter1-Inner1
outter1.foo(inner2) //  Outter1-Inner2

Summary

  • Scala 中非静态内部类属于外部类的对象而非外部类本身,而其它三种语言中则属于外部类
  • Scala 中可以定义外部类this的别名
  • Scala 与 Kotlin 没有静态外部类
  • Groovy 中内部类使用方法与 Java 基本一致,只是创建内部类语法的方式不一样
  • Kotlin 使用 this 的语法和其它三种语言都不一样

文章源码见 https://github.com/SidneyXu/JGSK 仓库的 _17_innerclass 小节

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

推荐阅读更多精彩内容

  • 1. Java基础部分 基础部分的顺序:基本语法,类相关的语法,内部类的语法,继承相关的语法,异常的语法,线程的语...
    子非鱼_t_阅读 31,577评论 18 399
  • 前言 人生苦多,快来 Kotlin ,快速学习Kotlin! 什么是Kotlin? Kotlin 是种静态类型编程...
    任半生嚣狂阅读 26,142评论 9 118
  • Java 内部类 分四种:成员内部类、局部内部类、静态内部类和匿名内部类。 1、成员内部类: 即作为外部类的一个成...
    ikaroskun阅读 1,219评论 0 13
  • 一:java概述:1,JDK:Java Development Kit,java的开发和运行环境,java的开发工...
    ZaneInTheSun阅读 2,629评论 0 11
  • 从前觉得18岁很远,一转眼已过5年了! 去年的生日,在杭州姐姐给过了一次,在无锡哥哥给过了一次,在家父母给过了一次...
    婚礼主持人诺言阅读 486评论 0 0