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逻辑了
    }
}
  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

_小马快跑_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值