数据类

    编译器自动从主构造函数中声明的所有属性导出以下成员:

    • equals()/hashCode() 对;
    • toString() 格式是 "User(name=John, age=42)"
    • 按声明顺序对应于所有属性;
    • copy() 函数(见下文)。

    为了确保生成的代码的一致性以及有意义的行为,数据类必须满足以下要求:

    • 主构造函数需要至少有一个参数;
    • 主构造函数的所有参数需要标记为 valvar
    • (在1.1之前)数据类只能实现接口。

    此外,成员生成遵循关于成员继承的这些规则:

    • 如果在数据类体中有显式实现 equals()hashCode() 或者 toString(),或者这些函数在父类中有 final 实现,那么不会生成这些函数,而会使用现有函数;
    • 如果超类型具有 opencomponentN() 函数并且返回兼容的类型, 那么会为数据类生成相应的函数,并覆盖超类的实现。如果超类型的这些函数由于签名不兼容或者是 final 而导致无法覆盖,那么会报错;
    • 从一个已具 copy(……) 函数且签名匹配的类型派生一个数据类在 Kotlin 1.2 中已弃用,并且在 Kotlin 1.3 中已禁用。
    • 不允许为 以及 copy() 函数提供显式实现。

    在 JVM 中,如果生成的类需要含有一个无参的构造函数,则所有的属性必须指定默认值。 (参见构造函数)。

    1. data class User(val name: String = "", val age: Int = 0)

    请注意,对于那些自动生成的函数,编译器只使用在主构造函数内部定义的属性。如需在生成的实现中排除一个属性,请将其声明在类体中:

    toString()equals()hashCode() 以及 copy() 的实现中只会用到 name 属性,并且只有一个 component 函数 component1()。虽然两个 Person 对象可以有不同的年龄,但它们会视为相等。

    1. data class Person(val name: String) {
    2. var age: Int = 0
    3. }
    4. fun main() {
    5. val person1 = Person("John")
    6. val person2 = Person("John")
    7. person1.age = 10
    8. person2.age = 20
    9. //sampleEnd
    10. println("person1 == person2: ${person1 == person2}")
    11. println("person1 with age ${person1.age}: ${person1}")
    12. println("person2 with age ${person2.age}: ${person2}")
    13. }

    这让我们可以写:

    1. val jack = User(name = "Jack", age = 1)

    为数据类生成的 Component 函数 使它们可在中使用:

    标准库提供了 与 Triple。尽管在很多情况下具名数据类是更好的设计选择, 因为它们通过为属性提供有意义的名称使代码更具可读性。