【对比Java学Kotlin】数据类

2023-11-15

我们在 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
}
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

【对比Java学Kotlin】数据类 的相关文章

  • 如何更改 JComboBox 下拉列表的宽度?

    我有一个可编辑的JComboBox其中包含单个字母值的列表 因此 组合框非常小 每个字母都有特殊的含义 对于很少使用的字母 有时用户并不清楚 因此我创建了一个自定义ListCellRenderer显示下拉列表中每个字母的含义 不幸的是 这个
  • 逐行读取 JTextPane

    有没有办法读取a的内容JTextPane逐行 很像 BufferedReader 吗 Element root textPane getDocument getDefaultRootElement 获得根元素后 您可以检查存在多少个子元素
  • Spring MVC 中的 CSRF(跨站请求伪造)保护

    我对春季的 CSRF 跨站请求伪造 保护有点困惑 不 我有我的 jsp 我的控制器和一个 Web 服务 我想要做的是在 Web 服务级别验证令牌 如果令牌匹配 则运行 Web 服务 在我的例子中执行数据库插入 JSP file
  • SimpleDateFormat 无法正确处理 DD

    我正在尝试获得这样的格式 2013 06 15 17 45 我在代码中执行以下操作 Date d new Date SimpleDateFormat ft new SimpleDateFormat YYYY MM DD HH mm Stri
  • AES BadPaddingException

    如果我使用错误的密钥或错误的盐进行解密 则会引发 BadPaddingException 我希望返回一个不正确的字符串 doFinal 导致解密方法出现异常 信息 This is just an example Unfug S F V s
  • Hibernate、MySQL 视图和 hibernate.hbm2ddl.auto = 验证

    我可以在 Hibernate 中使用 MySQL 视图 将它们视为表 即 该实体与为表创建的实体没有什么不同 但是 当 Hibernate 设置为验证模型时 我的应用程序将不会部署 因为它找不到视图 因为它假设它是一个表 是否可以在启用部署
  • 在Tomcat中设置环境变量TESSDATA_PREFIX

    我们正在使用名为 Tess4J 的 Tesseract OCR Java 库 如果作为独立应用程序运行 它可以正常工作 它需要一个名为 TESSDATA PREFIX 的变量 其中包含 tessdata 配置和其他字符集相关文件 它也可以与
  • JavaFX 动画使用循环?

    我正在尝试制作一款类似太空侵略者的游戏 我画了一个正方形 我想通过使用循环逐步向下移动它thread sleep 然而 正方形立即被绘制出来 我知道有可以使用的动画路径 但我想保持低水平并仅使用坐标系 有没有办法使用这样的循环来制作时间轴动
  • 不想保留一对一的实体

    假设我有两节课Employee and Department In Employee我已经写了 OneToOne fetch FetchType EAGER cascade CascadeType ALL JoinColumn name d
  • APACHE POI 从 Java 中的 Excel 获取精确的字体颜色

    在 Excel 工作表中 如何使用 Java 中的 Apache POI 获取准确的字体颜色值 我试图通过使用来获取字体颜色 org apache poi ss usermodel Font f book getFontAt style g
  • 如何使 RSACryptoServiceProvider 在没有填充(nopadding)的情况下工作?

    我需要使 C 应用程序与 Java 应用程序兼容 Java 应用程序使用Cipher getInstance RSA ECB nopadding 初始化器使密码 ECB 和无填充 但是 在 C 中 您有 2 个填充选项 OAEP 填充或 P
  • JavaFX 8 默认消息图标

    随着 JavaFX 的最近几次更新 我们收到了警报 我想获取消息的默认图标 错误 警告 在Swing中 我可以通过一些方式获取L F消息图标UIManager的属性 如何在 JavaFX 中获取消息的默认图标 它们是包含在属性中 还是由 C
  • 从字符串中提取文本 Java

    使用此字符串 ADACADABRA 如何从java中的字符串 ADACADABRA 中提取 CADA 以及如何提取 和 之间的id从下面的链接 http www youtube nocookie com embed zaaU9lJ34c5
  • 解析 SWIG 接口文件的结构属性

    这是我不久前问过的问题的延续 为通过参数返回的函数创建类型映射 https stackoverflow com questions 12793973 create a typemap for a function that returns
  • JSF - 实施受限页面过滤器

    我正在关注 BalusC 的回答JSF 2 0 如何获取在浏览器地址栏中输入的 URL https stackoverflow com questions 4105263 jsf 2 0 how to get the url that is
  • JavaFX 中的 MVC 模式与场景生成器

    我是 JavaFX 新手 根据我当前的设置 正在努力创建合适的 MVC 架构 我使用 Scene Builder 单击了一个 UI 并指定了一个 Controller 类 Startup public class Portal extend
  • Hibernate 命名查询使用 Like 和 % % 运算符?

    在我的 Hibernate JPA 示例代码中 public List
  • Java:如何检测(并更改?)System.console 的编码?

    我有一个在控制台上运行的程序 其变音符号和其他特殊字符在 Mac 上以 的形式输出 这是一个简单的测试程序 public static void main String args System out println h h System
  • 如何获取 res.drawable 文件夹的路径来复制文件?

    我正在编写我的应用程序AndroidStudio 我的里面有gif文件drawable gifs文件夹 我希望将该文件复制到MediaStore Images Media单击按钮后的文件夹 目前 即使使用发布的一些答案 我也无法获取我的 g
  • selenium 没有找到合适的方法,直到(ExpectedCondition)

    这是有线的问题 我导入的项目运行 100 几个月前 今天我已将其与依赖项一起导入 但存在问题WebDriverWait 这是我的代码 WebDriverWait driverWait new WebDriverWait driver 100

随机推荐