我们经常创建一个类,只能持有数据。 在这样一个类中,一些标准功能通常是从数据中机械推导出来的。 在Kotlin中,这被称为数据类,并被标记为数据:
data class User(val name: String, val age: Int)
编译器自动从主构造函数中声明的所有属性导出以下成员:
equals()
/hashCode()
函数对toString()
函数,它由”User(name=John, age=42)
“构成- componentN()函数对应于属性的声明顺序
copy()
函数(见下文)
如果这些函数中的任何一个在类体中显式定义或继承自其基类型,则不会生成该函数。
为了确保生成的代码的一致性和有意义的行为,数据类必须满足以下要求:
- 主构造函数需要至少有一个参数;
- 主构造函数的所有参数需要标记为
val
或var
; - 数据类不能是抽象、开放、密封或者内部的;
- 数据类只能实现接口(在1.1之前)。
自 1.1 起,数据类可以扩展其他类(示例请参见密封类)。
在 JVM 中,如果生成的类需要含有一个无参的构造函数,则所有的属性必须指定默认值。 (参见构造函数)。
data class User(val name: String = "", val age: Int = 0)
复制
在很多情况下,我们需要复制一个对象改变它的一些属性,但其余部分保持不变。 copy() 函数就是为此而生成。对于上文的 User 类,其实现会类似下面这样:
fun copy(name: String = this.name, age: Int = this.age) = User(name, age)
这也可以写成 -
val jack = User(name = "Jack", age = 1)
val olderJack = jack.copy(age = 2)
数据类和解构声明
为数据类生成的 Component
函数 使它们可在解构声明中使用:
val jane = User("Jane", 35)
val (name, age) = jane
println("$name, $age years of age") // 输出 "Jane, 35 years of age"
标准数据类
标准库提供了 Pair 和 Triple。尽管在很多情况下命名数据类是更好的设计选择, 因为它们通过为属性提供有意义的名称使代码更具可读性。