Kotlin data数据类、copy()函数、sealed密封类

简介: Kotlin data数据类、copy()函数、sealed密封类使用

data数据类

data class ModelA(
    val name: String = "",
    var age: Int = 10,
    var grade: Int = 6,
)
  • 主构造函数需要至少有一个参数
  • 主构造函数的所有参数需要标记为 valvar
  • 数据类不能被abstractopensealed或者internal修饰;

转换成Java类:

public final class ModelA {
   @NotNull
   private final String name;
   private int age;
   private int grade;

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

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

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

   public final int getGrade() {
      return this.grade;
   }

   public final void setGrade(int var1) {
      this.grade = var1;
   }

   public ModelA(@NotNull String name, int age, int grade) {
      Intrinsics.checkNotNullParameter(name, "name");
      super();
      this.name = name;
      this.age = age;
      this.grade = grade;
   }

   // $FF: synthetic method
   public ModelA(String var1, int var2, int var3, int var4, DefaultConstructorMarker var5) {
      if ((var4 & 1) != 0) {
         var1 = "";
      }

      if ((var4 & 2) != 0) {
         var2 = 10;
      }

      if ((var4 & 4) != 0) {
         var3 = 6;
      }

      this(var1, var2, var3);
   }

   public ModelA() {
      this((String)null, 0, 0, 7, (DefaultConstructorMarker)null);
   }

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

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

   public final int component3() {
      return this.grade;
   }

   @NotNull
   public final ModelA copy(@NotNull String name, int age, int grade) {
      Intrinsics.checkNotNullParameter(name, "name");
      return new ModelA(name, age, grade);
   }

   // $FF: synthetic method
   public static ModelA copy$default(ModelA var0, String var1, int var2, int 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.grade;
      }

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

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

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

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

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

可以看到data数据类帮我们生成了equals()、hashCode()、toString()、copy()函数。

copy()函数

当需要复制一个对象,并需要改变部分属性值时,copy()函数为此而生。

val modelA = ModelA(name = "A", age = 10, grade = 1)
val modelB = modelA.copy(name = "B")

log("modelA:$modelA,hashCode:${modelA.hashCode()}")
log("modelB:$modelB,hashCode:${modelB.hashCode()}")
//执行结果:
modelA:ModelA(name=A, age=10, grade=1),hashCode:62776
modelB:ModelA(name=B, age=10, grade=1),hashCode:63737

如果copy()函数中改变了对象中的属性,会通过new重新生成一个新对象,示例中通过结果中不同的hashCode值即可看到;而如果只是调用copy()函数,并未改变对象的属性值时,通过实验发现两者的hashCode值并未改变。

sealed密封类

sealed密封类用来表示受限的类继承结构:当一个值为有限几种的类型、而不能有任何其他类型时

可以将密封类对比枚举类:枚举常量只存在一个实例,而密封类的一个子类可以有可包含状态的多个实例。

sealed class Async<out T> {
    object Loading : Async<Nothing>()
    data class Success<out T>(val data: T) : Async<T>()
}

转换成Java类:

//注意看,这里是abstract抽象类
public abstract class Async {
   private Async() {
   }

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

   public static final class Loading extends Async {
      @NotNull
      public static final Async.Loading INSTANCE;

      private Loading() {
         super((DefaultConstructorMarker)null);
      }

      static {
         Async.Loading var0 = new Async.Loading();
         INSTANCE = var0;
      }
   }

   public static final class Success extends Async {
      private final Object data;

      public final Object getData() {
         return this.data;
      }

      public Success(Object data) {
         super((DefaultConstructorMarker)null);
         this.data = data;
      }

      public final Object component1() {
         return this.data;
      }

      @NotNull
      public final Async.Success copy(Object data) {
         return new Async.Success(data);
      }

      // $FF: synthetic method
      public static Async.Success copy$default(Async.Success var0, Object var1, int var2, Object var3) {
         if ((var2 & 1) != 0) {
            var1 = var0.data;
         }

         return var0.copy(var1);
      }

      @NotNull
      public String toString() {
         return "Success(data=" + this.data + ")";
      }

      public int hashCode() {
         Object var10000 = this.data;
         return var10000 != null ? var10000.hashCode() : 0;
      }

      public boolean equals(@Nullable Object var1) {
         if (this != var1) {
            if (var1 instanceof Async.Success) {
               Async.Success var2 = (Async.Success)var1;
               if (Intrinsics.areEqual(this.data, var2.data)) {
                  return true;
               }
            }

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

可以看到转换成Java类之后,sealed密封类自身是abstract抽象类,不能直接进行实例化。
sealed密封类的典型用法即是在when(){}中,当传入的是密封类时,不用再写else分支了,如:

fun processResult(result: Async<String>) {
    when (result) {
        is Async.Loading -> {
            // do something
        }
        is Async.Success -> {
            //do something
        }
        //这里不用再写else逻辑了
    }
}
相关文章
|
1月前
|
Kotlin
Kotlin教程笔记(20) - 枚举与密封类
Kotlin教程笔记(20) - 枚举与密封类
42 8
|
1月前
|
安全 Kotlin
Kotlin教程笔记(23) -作用域函数
Kotlin教程笔记(23) -作用域函数
124 6
|
1月前
|
Kotlin
Kotlin教程笔记(21) -高阶函数与函数引用
Kotlin教程笔记(21) -高阶函数与函数引用
40 6
|
1月前
|
Java Kotlin
Kotlin教程笔记(13) - 类及成员的可见性
Kotlin教程笔记(13) - 类及成员的可见性
42 3
|
1月前
|
存储 前端开发 Java
Kotlin教程笔记(18) - 数据类
Kotlin教程笔记(18) - 数据类
|
2月前
|
IDE 开发工具 Kotlin
Kotlin教程笔记(6) - 函数与Lambda表达式
Kotlin教程笔记(6) - 函数与Lambda表达式
58 1
|
2月前
|
Java 开发者 Kotlin
Kotlin教程笔记(2) - 类与构造器
Kotlin教程笔记(2) - 类与构造器
29 1
|
2月前
|
Java Kotlin
​ Kotlin教程笔记(13) - 类及成员的可见性
​ Kotlin教程笔记(13) - 类及成员的可见性
|
2月前
|
数据安全/隐私保护 Kotlin
Kotlin教程笔记(7) - 类成员
Kotlin教程笔记(7) - 类成员
|
3月前
|
JSON 调度 数据库
Android面试之5个Kotlin深度面试题:协程、密封类和高阶函数
本文首发于公众号“AntDream”,欢迎微信搜索“AntDream”或扫描文章底部二维码关注,和我一起每天进步一点点。文章详细解析了Kotlin中的协程、扩展函数、高阶函数、密封类及`inline`和`reified`关键字在Android开发中的应用,帮助读者更好地理解和使用这些特性。
47 1