对比Java学Kotlin数据类

Posted 陈蒙_

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了对比Java学Kotlin数据类相关的知识,希望对你有一定的参考价值。

我们在 Java 里面会创建一些专门用于盛放数据的类,比如各种以 Bean、Model 作为后缀结尾的类。

这些类的成员变量通常是各种类型的数据,成员函数是 setter 和 getter。或者偷懒的同学直接把成员变量的可见性设置为 public,连 setter 和 getter 都省了。

虽然我们可以省掉 setter&getter 这些模板代码,toString()、equals() 、hashCode() 和 copy() 这些方法还是需要手动实现的。

Kotlin 专门设置了数据类来简化这些模板代码,让代码更加简洁。

我们以 Puppy 这个类为例:

// 定义数据类
data class Puppy(
    val name: String,
    val age: Int,
    val cuteness: Int,
) {
    var breed: String? = null
}

// 使用方法
val tofu = Puppy(name = "Tofu", age = 1, cuteness = Int.MAX_VALUE)
val taco = Puppy(name = "Taco", age = 2)

fun play() {
    val anotherTaco = taco.copy(name = "Tofu2") // 其余成员变量使用既有值
}

我们来看反编译后的 Java 代码:

public final class Puppy {
   @Nullable
   private String breed;
   @NotNull
   private final String name;
   private final int age;
   @NotNull
   private final String cuteness;

   @Nullable
   public final String getBreed() {
      return this.breed;
   }

   public final void setBreed(@Nullable String var1) {
      this.breed = var1;
   }

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

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

   @NotNull
   public final String getCuteness() {
      return this.cuteness;
   }

   public Puppy(@NotNull String name, int age, @NotNull String cuteness) {
      Intrinsics.checkParameterIsNotNull(name, "name");
      Intrinsics.checkParameterIsNotNull(cuteness, "cuteness");
      super();
      this.name = name;
      this.age = age;
      this.cuteness = cuteness;
   }

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

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

   @NotNull
   public final String component3() {
      return this.cuteness;
   }

   @NotNull
   public final Puppy copy(@NotNull String name, int age, @NotNull String cuteness) {
      Intrinsics.checkParameterIsNotNull(name, "name");
      Intrinsics.checkParameterIsNotNull(cuteness, "cuteness");
      return new Puppy(name, age, cuteness);
   }

   // $FF: synthetic method
   public static Puppy copy$default(Puppy var0, String var1, int var2, String var3, int var4, Object var5) {
      if ((var4 & 1) != 0) {
         var1 = var0.name;
      }

      if ((var4 & 2) != 0) {
         var2 = var0.age;
      }

      if ((var4 & 4) != 0) {
         var3 = var0.cuteness;
      }

      return var0.copy(var1, var2, var3);
   }

   @NotNull
   public String toString() {
      return "Puppy(name=" + this.name + ", age=" + this.age + ", cuteness=" + this.cuteness + ")";
   }

   public int hashCode() {
      String var10000 = this.name;
      int var1 = ((var10000 != null ? var10000.hashCode() : 0) * 31 + this.age) * 31;
      String var10001 = this.cuteness;
      return var1 + (var10001 != null ? var10001.hashCode() : 0);
   }

   public boolean equals(@Nullable Object var1) {
      if (this != var1) {
         if (var1 instanceof Puppy) {
            Puppy var2 = (Puppy)var1;
            if (Intrinsics.areEqual(this.name, var2.name) && this.age == var2.age && Intrinsics.areEqual(this.cuteness, var2.cuteness)) {
               return true;
            }
         }

         return false;
      } else {
         return true;
      }
   }
}

可以看到,data class 为每个属性:

  • 自动生成了 getter&setter
  • 自动使用主构造函数中的属性生成 equals() & hashCode() & copy() & toString()方法

简直不要太方便!终于可以摆脱烦人的模板代码了!

同时,我们需要注意:

  • 数据类不能是 abstract, open, sealed, 或者 inner 的,否则会报错
  • 数据类可以继承其他的类或者接口或者抽象类,但是不能继承其他的数据类
  • 数据类的主构造方法中只能是val/var类型的成员变量,只有主构造方法中的成员变量才会在被用于生成equals() & hashCode() & copy() & toString()方法,类体内的成员变量则不会
  • 数据类主构造方法中的成员变量建议使用 val 类型,因为当数据类的实例作为 HashMap 等容器的 key 使用时,var 类型的变量的改变会导致我们得到错误的 value

Koltin 内置了 Pair 和 Triple 这两个数据类,分别用于承载两个和三个成员变量的场景。

解构声明

在反编译的 Java 代码中有几个函数比较陌生:

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

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

   @NotNull
   public final String component3() {
      return this.cuteness;
   }

什么是 component?作用是啥?在什么场景下用?
这是给解构声明(Destructuring Declaration)用的,主要是为了实现一次性的从一个对象中解析出多个变量,比如:

val tofu = Puppy(name = "Tofu", age = 1, cuteness = Int.MAX_VALUE)

fun play() {
    val (name, age) = tofu // name=tofu.name, age=tofu.age
    val (name1, _, cuteness1) = tofu // name1=tofu.name, cuteness1=tofu.cuteness
}

可以看出,右值表达式的顺序是跟左值对象的 componentN() 对应的,如果序列中间的某个变量不需要,可将其声明为下划线_,则自动会被忽略。解构声明等价于:

val name = tofu.component1()
val age = tofu.component2()

此外,Kotlin 的 Map 函数也实现了结构声明的功能,比如我们在遍历 Map 函数是:

val map = mapOf<String, Int>()
for ((k, v) in map) {
    print("k: $k, v: $v")
}

那普通的类可以实现解构声明吗?实际上是可以的,结构功能并未数据类或者 Map 专有,而是只要实现操作符 componentN() 即可。比如我们有一个普通的类 Puppy:

class Puppy(
    val name: String,
    val age: Int,
    val cuteness: Int = 11
) {
    var breed: String? = null

    operator fun component1() = name
    operator fun component2() = age
    operator fun component3() = cuteness
    operator fun component4() = breed
}

// 使用解构声明
val tofu = Puppy(name = "Tofu", age = 1, cuteness = Int.MAX_VALUE)

fun play() {
    val (name, age) = tofu // name=tofu.name, age=tofu.age
    val (name1, _, cuteness1) = tofu // name1=tofu.name, cuteness1=tofu.cuteness
}

以上是关于对比Java学Kotlin数据类的主要内容,如果未能解决你的问题,请参考以下文章

对比Java学Kotlin数据类

对比Java学Kotlin官方文档目录

对比Java学Kotlin密封类

对比Java学Kotlin密封类

对比Java学Kotlin密封类

对比Java学Kotlin嵌套类和内部类