【Kotlin】类的初始化 ③ ( init 初始化块 | 初始化顺序 : 主构造函数属性赋值 -> 类属性赋值 -> init 初始化块代码 -> 次构造函数代码 )

2023-03-30 18:52:53 浏览数 (1)

文章目录

  • 一、init 初始化块
  • 二、初始化顺序

一、init 初始化块


在 Kotlin 类中 , 可以定义 init 初始化块 , 在其中可以为 变量赋值 , 执行一些检查相关的代码 , 该 init 初始化块在 创建类实例对象 时执行 ;

代码示例 : 在下面的代码中的 init 初始化块 中 , 对 name 属性进行了修改 , 检查了 age 属性是否合法 ;

代码语言:javascript复制
class Hello(
    // 主构造函数, 直接在主构造函数中定义属性
    var name: String,
    var age: Int
){
    init {
        // 将 name 属性首字母大写
        name = name.capitalize()

        // 检查 age 是否合法
        // 如果不符合要求, 则抛出异常
        require(age > 0) {
            println("年龄必须大于 0")
        }
    }
}

fun main() {
    var hello = Hello("tom", 18)
    println(hello.name   " , "   hello.age)

    var hello2 = Hello("jerry", -1)
}

执行结果 :

代码语言:javascript复制
Tom , 18
年龄必须大于 0
Exception in thread "main" java.lang.IllegalArgumentException: kotlin.Unit
	at Hello.<init>(Hello.kt:11)
	at HelloKt.main(Hello.kt:21)
	at HelloKt.main(Hello.kt)

二、初始化顺序


Kotlin 类 对象在实例化 时会执行一系列的 初始化操作 , 这些操作按照如下顺序执行 :

  • 主构造函数 中属性赋值
  • 类中的属性赋值
  • init 初始化块 中的代码执行
  • 次构造函数 中的代码执行

代码示例 : 通过下面的代码分析 Kotlin 实例对象 各种初始化操作的 初始化顺序 ;

代码语言:javascript复制
class Hello(
    // 主构造函数, 直接在主构造函数中定义属性
    var name: String,
    // 该值是临时变量, 为 age 属性赋值
    _age: Int
){
    // 类中的属性
    var age = _age
    var type = "老鼠"
    var gender: String

    init {
        println("init 初始化块开始执行")
        gender = "男"
    }

    constructor(_age: Int): this("Tom", _age) {
        println("次构造函数开始执行")
        type = "猫"
    }
}

fun main() {
    var hello = Hello(18)
    println(hello.name   " , "   hello.age)
}

执行结果 :

代码语言:javascript复制
init 初始化块开始执行
次构造函数开始执行
Tom , 18

从上述执行结果上看 , 可以知道先执行 init 初始化块 , 然后执行 次构造函数 ;

查看 Kotlin 字节码 , 并将其反编译回 Java 代码 , 结果如下 :

代码语言:javascript复制
// HelloKt.java
import kotlin.Metadata;

@Metadata(
   mv = {1, 4, 2},
   bv = {1, 0, 3},
   k = 2,
   d1 = {"u0000bnu0000nu0002u0010u0002nu0000u001au0006u0010u0000u001au00020u0001¨u0006u0002"},
   d2 = {"main", "", "KotlinDemo"}
)
public final class HelloKt {
   public static final void main() {
      Hello hello = new Hello(18);
      String var1 = hello.getName()   " , "   hello.getAge();
      boolean var2 = false;
      System.out.println(var1);
   }

   // $FF: synthetic method
   public static void main(String[] var0) {
      main();
   }
}
// Hello.java
import kotlin.Metadata;
import kotlin.jvm.internal.Intrinsics;
import org.jetbrains.annotations.NotNull;

@Metadata(
   mv = {1, 4, 2},
   bv = {1, 0, 3},
   k = 1,
   d1 = {"u0000u001anu0002u0018u0002nu0002u0010u0000nu0000nu0002u0010bnu0002bu0002nu0002u0010u000enu0002bu0010u0018u00002u00020u0001Bu000fbu0016u0012u0006u0010u0002u001au00020u0003¢u0006u0002u0010u0004Bu0015u0012u0006u0010u0005u001au00020u0006u0012u0006u0010u0002u001au00020u0003¢u0006u0002u0010u0007Ru001au0010bu001au00020u0003Xu0086u000e¢u0006u000enu0000u001au0004btu0010n"u0004bu000bu0010u0004Ru001au0010fu001au00020u0006Xu0086u000e¢u0006u000enu0000u001au0004bru0010u000e"u0004bu000fu0010u0010Ru001au0010u0005u001au00020u0006Xu0086u000e¢u0006u000enu0000u001au0004bu0011u0010u000e"u0004bu0012u0010u0010Ru001au0010u0013u001au00020u0006Xu0086u000e¢u0006u000enu0000u001au0004bu0014u0010u000e"u0004bu0015u0010u0010¨u0006u0016"},
   d2 = {"LHello;", "", "_age", "", "(I)V", "name", "", "(Ljava/lang/String;I)V", "age", "getAge", "()I", "setAge", "gender", "getGender", "()Ljava/lang/String;", "setGender", "(Ljava/lang/String;)V", "getName", "setName", "type", "getType", "setType", "KotlinDemo"}
)
public final class Hello {
   private int age;
   @NotNull
   private String type;
   @NotNull
   private String gender;
   @NotNull
   private String name;

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

   public final void setAge(int var1) {
      this.age = var1;
   }

   @NotNull
   public final String getType() {
      return this.type;
   }

   public final void setType(@NotNull String var1) {
      Intrinsics.checkNotNullParameter(var1, "<set-?>");
      this.type = var1;
   }

   @NotNull
   public final String getGender() {
      return this.gender;
   }

   public final void setGender(@NotNull String var1) {
      Intrinsics.checkNotNullParameter(var1, "<set-?>");
      this.gender = var1;
   }

   @NotNull
   public final String getName() {
      return this.name;
   }

   public final void setName(@NotNull String var1) {
      Intrinsics.checkNotNullParameter(var1, "<set-?>");
      this.name = var1;
   }

   public Hello(@NotNull String name, int _age) {
      Intrinsics.checkNotNullParameter(name, "name");
      super();
      this.name = name;
      this.age = _age;
      this.type = "老鼠";
      String var3 = "init 初始化块开始执行";
      boolean var4 = false;
      System.out.println(var3);
      this.gender = "男";
   }

   public Hello(int _age) {
      this("Tom", _age);
      String var2 = "次构造函数开始执行";
      boolean var3 = false;
      System.out.println(var2);
      this.type = "猫";
   }
}

重点分析 构造函数 :

代码语言:javascript复制
   public Hello(@NotNull String name, int _age) {
      Intrinsics.checkNotNullParameter(name, "name");
      super();
      this.name = name;
      this.age = _age;
      this.type = "老鼠";
      String var3 = "init 初始化块开始执行";
      boolean var4 = false;
      System.out.println(var3);
      this.gender = "男";
   }

在上述 构造函数中 :

首先 , 为 name 属性赋值 , 这是在 主构造函数 中完成的操作 ;

然后 , 为 agetype 属性赋值 , 这是在 类 中的 age 属性进行的赋值 , 使用的是 主构造函数 中的临时变量 ;

最后 , 为 gender 赋值 , 这是在 init 初始化块 中进行的赋值 ;

然后分析 次构造函数 , 在 如下的 次构造函数的代码中 , 先执行了 主构造函数 , 然后才为 type 属性赋值 , 这是在次构造函数中执行的 , 这是最后执行的代码 ;

代码语言:javascript复制
   public Hello(int _age) {
      this("Tom", _age);
      String var2 = "次构造函数开始执行";
      boolean var3 = false;
      System.out.println(var2);
      this.type = "猫";
   }

因此得到了上述初始化操作的执行顺序 : 主构造函数属性赋值 -> 类属性赋值 -> init 初始化块代码 -> 次构造函数代码

0 人点赞