kotlin之object,companion object,内部class和inner class之间的区别

最近开始使用kotlin写项目了,这几天一直对object,companion object,内部类inner class搞得有点迷糊

现在抽空,理清一下他们之间区别

我定义了一个class A
然后再class A内部分别定义了class B,object C,companion object D,inner class E四中不同的类
每个类中包含一个字段和一个函数

/**
 * Copyright:AndroidInterview
 * Author: liyang <br>
 * Date:2019-11-12 14:29<br>
 * Desc: <br>
 */
class A  {
    var sex = "男"
    val age = 11
    val a1:String="nan"
    val a: String
        get() {
            return when (sex) {
                "男" -> "man"
                else -> "women"
            }
        }

    fun a() {
        //class B
        //在A类中调用class B的b方法
        val b = B()
        b.b()
        b.b
        
        //inner class E
        //在A类中调用inner class E的e变量和e()方法
        val e=E()
        e.e
        e.e()

        //object C
        //在A类中调用object C的c()方法和c变量
        C.c()
        C.c

        //companion object D
        //可以直接在A类中调用伴生类的d()方法和d变量
        d()
        d
    }

    class B {
        val b = "b"
        fun b() {
            println("b() is invoked")
        }
    }

    object C {
       const val c = "c"
        fun c() {
            println("c() is invoked")
        }
    }


    companion object D {
        val d = "d"

        fun d() {
            println("d() is invoked")
        }
    }

    inner class E {
        val e = "e"
        fun e() {
            println("e() is invoked")
        }
    }


}

接下来在main方法中调用它们,看看调用的时候有什么不同

/**
 * Copyright:AndroidInterview
 * Author: liyang <br>
 * Date:2019-11-12 14:29<br>
 * Desc: <br>
 */
open class Outer {
    companion object {
        @JvmStatic
        fun main(args: Array<String>) {
            val a = A()
            a.a
            a.a()

            //内部类
            val b = A.B()
            println(b.b)
            b.b()

            //inner class
            val e = a.E()
            println(e.e)
            e.e()

            //object 
            println(A.C.c)
            A.C.c()

            //companion object
            println(A.d)
            A.d()
        }
    }
}
内部类类型 在A类内部调用 在A类外部调用
object C C.c(),C.c A.C.c,A.C.c()
companion object D 可以直接使用 d,d() A.d,A.d()
class B 实例化B对象val b = B(); b.b() ; b.b 实例化B对象val b = A.B(); b.b() ; b.b
inner class E 实例化E对象val e = E(); e.e() ; e.e 通过外部类的实例a来实例化E对象val e = a.E(); e.e() ; e.e

为什么这样?反编译看看源码就知道了

A.decompiled.java

import kotlin.Metadata;
import kotlin.jvm.internal.DefaultConstructorMarker;
import kotlin.jvm.internal.Intrinsics;
import org.jetbrains.annotations.NotNull;

@Metadata(
   mv = {1, 1, 15},
   bv = {1, 0, 3},
   k = 1,
   d1 = {"\u0000$\n\u0002\u0018\u0002\n\u0002\u0010\u0000\n\u0002\b\u0002\n\u0002\u0010\u000e\n\u0002\b\u0005\n\u0002\u0010\b\n\u0002\b\u0006\n\u0002\u0010\u0002\n\u0002\b\u0005\u0018\u0000 \u00142\u00020\u0001:\u0004\u0012\u0013\u0014\u0015B\u0005¢\u0006\u0002\u0010\u0002J\u0006\u0010\u0003\u001a\u00020\u0011R\u0011\u0010\u0003\u001a\u00020\u00048F¢\u0006\u0006\u001a\u0004\b\u0005\u0010\u0006R\u0014\u0010\u0007\u001a\u00020\u0004X\u0086D¢\u0006\b\n\u0000\u001a\u0004\b\b\u0010\u0006R\u0014\u0010\t\u001a\u00020\nX\u0086D¢\u0006\b\n\u0000\u001a\u0004\b\u000b\u0010\fR\u001a\u0010\r\u001a\u00020\u0004X\u0086\u000e¢\u0006\u000e\n\u0000\u001a\u0004\b\u000e\u0010\u0006\"\u0004\b\u000f\u0010\u0010¨\u0006\u0016"},
   d2 = {"LA;", "", "()V", "a", "", "getA", "()Ljava/lang/String;", "a1", "getA1", "age", "", "getAge", "()I", "sex", "getSex", "setSex", "(Ljava/lang/String;)V", "", "B", "C", "D", "E", "baselib"}
)
public final class A {
   @NotNull
   private String sex = "男";
   private final int age = 11;
   @NotNull
   private final String a1 = "nan";
   @NotNull
   private static final String d = "d";
   public static final A.D D = new A.D((DefaultConstructorMarker)null);

   @NotNull
   public final String getSex() {
      return this.sex;
   }

   public final void setSex(@NotNull String var1) {
      Intrinsics.checkParameterIsNotNull(var1, "<set-?>");
      this.sex = var1;
   }

   public final int getAge() {
      return this.age;
   }

   @NotNull
   public final String getA1() {
      return this.a1;
   }

   @NotNull
   public final String getA() {
      String var1 = this.sex;
      String var10000;
      switch(var1.hashCode()) {
      case 30007:
         if (var1.equals("男")) {
            var10000 = "man";
            break;
         }
      default:
         var10000 = "women";
      }

      return var10000;
   }

   public final void a() {
      A.B b = new A.B();
      b.b();
      b.getB();
      A.C.INSTANCE.c();
      D.d();
      String var10000 = d;
   }

   @Metadata(
      mv = {1, 1, 15},
      bv = {1, 0, 3},
      k = 1,
      d1 = {"\u0000\u001a\n\u0002\u0018\u0002\n\u0002\u0010\u0000\n\u0002\b\u0002\n\u0002\u0010\u000e\n\u0002\b\u0002\n\u0002\u0010\u0002\n\u0000\u0018\u00002\u00020\u0001B\u0005¢\u0006\u0002\u0010\u0002J\u0006\u0010\u0003\u001a\u00020\u0007R\u0014\u0010\u0003\u001a\u00020\u0004X\u0086D¢\u0006\b\n\u0000\u001a\u0004\b\u0005\u0010\u0006¨\u0006\b"},
      d2 = {"LA$B;", "", "()V", "b", "", "getB", "()Ljava/lang/String;", "", "baselib"}
   )
   public static final class B {
      @NotNull
      private final String b = "b";

      @NotNull
      public final String getB() {
         return this.b;
      }

      public final void b() {
         String var1 = "b() is invoked";
         boolean var2 = false;
         System.out.println(var1);
      }
   }

   @Metadata(
      mv = {1, 1, 15},
      bv = {1, 0, 3},
      k = 1,
      d1 = {"\u0000\u0016\n\u0002\u0018\u0002\n\u0002\u0010\u0000\n\u0002\b\u0002\n\u0002\u0010\u000e\n\u0002\u0010\u0002\n\u0000\bÆ\u0002\u0018\u00002\u00020\u0001B\u0007\b\u0002¢\u0006\u0002\u0010\u0002J\u0006\u0010\u0003\u001a\u00020\u0005R\u000e\u0010\u0003\u001a\u00020\u0004X\u0086T¢\u0006\u0002\n\u0000¨\u0006\u0006"},
      d2 = {"LA$C;", "", "()V", "c", "", "", "baselib"}
   )
   public static final class C {
      @NotNull
      public static final String c = "c";
      public static final A.C INSTANCE;

      public final void c() {
         String var1 = "c() is invoked";
         boolean var2 = false;
         System.out.println(var1);
      }

      private C() {
      }

      static {
         A.C var0 = new A.C();
         INSTANCE = var0;
      }
   }

   @Metadata(
      mv = {1, 1, 15},
      bv = {1, 0, 3},
      k = 1,
      d1 = {"\u0000\u001a\n\u0002\u0018\u0002\n\u0002\u0010\u0000\n\u0002\b\u0002\n\u0002\u0010\u000e\n\u0002\b\u0002\n\u0002\u0010\u0002\n\u0000\b\u0086\u0004\u0018\u00002\u00020\u0001B\u0005¢\u0006\u0002\u0010\u0002J\u0006\u0010\u0003\u001a\u00020\u0007R\u0014\u0010\u0003\u001a\u00020\u0004X\u0086D¢\u0006\b\n\u0000\u001a\u0004\b\u0005\u0010\u0006¨\u0006\b"},
      d2 = {"LA$E;", "", "(LA;)V", "e", "", "getE", "()Ljava/lang/String;", "", "baselib"}
   )
   public final class E {
      @NotNull
      private final String e = "e";

      @NotNull
      public final String getE() {
         return this.e;
      }

      public final void e() {
         String var1 = "e() is invoked";
         boolean var2 = false;
         System.out.println(var1);
      }
   }

   @Metadata(
      mv = {1, 1, 15},
      bv = {1, 0, 3},
      k = 1,
      d1 = {"\u0000\u001a\n\u0002\u0018\u0002\n\u0002\u0010\u0000\n\u0002\b\u0002\n\u0002\u0010\u000e\n\u0002\b\u0002\n\u0002\u0010\u0002\n\u0000\b\u0086\u0003\u0018\u00002\u00020\u0001B\u0007\b\u0002¢\u0006\u0002\u0010\u0002J\u0006\u0010\u0003\u001a\u00020\u0007R\u0014\u0010\u0003\u001a\u00020\u0004X\u0086D¢\u0006\b\n\u0000\u001a\u0004\b\u0005\u0010\u0006¨\u0006\b"},
      d2 = {"LA$D;", "", "()V", "d", "", "getD", "()Ljava/lang/String;", "", "baselib"}
   )
   public static final class D {
      @NotNull
      public final String getD() {
         return A.d;
      }

      public final void d() {
         String var1 = "d() is invoked";
         boolean var2 = false;
         System.out.println(var1);
      }

      private D() {
      }

      // $FF: synthetic method
      public D(DefaultConstructorMarker $constructor_marker) {
         this();
      }
   }
}

通过查看decompile A.kt文件之后得到的A.decompiled.java源文件
可以得到以下几个重要信息

  • kotlin中只要不是被inner修饰的内部类,都是静态的!所以class B,object C,companion object D都是静态内部类
  • inner class E是非静态内部类,所以它会持有A的引用
  • object Ccompanion object D的构造函数都是private
  • 无论是object还是companion object中的变量都是静态常量

梳理下

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