数据类
我们经常创建一些只保存数据的类。在这些类中,一些标准函数往往是从数据机械推导而来的。在Kotlin中,这叫做数据类并标记为data:
data class User(val name:String,val age:Int)
编译器自动从主构造函数中声明的所有属性导出以下成员:
- equals()/hashCode()对;
-toString()格式是"User(name=John,age=42)";
-componentN()函数 按声明顺序对应于所有属性;
-copy()函数
为了确保生成的代码的一致性以及有意义的行为,数据类必须满足以下需求:
- 主构造函数需要至少有一个参数
-主构造函数的所有参数需要标记为val或者var;
-数据类不能是抽象,开放,密封或者内部的
此外,成员生成遵循关于成员继承的这些规则:
-如果在数据类体重有显示实现equals(),hashCode()或者toString(),或者这些函数在父类中有final实现,那么不会生成这些函数,而会使用现有函数;
-如果超类型具有open的componentN()函数并且返回兼容类型,那么会为数据类型生成相应的函数,并覆盖超类的实现。如果超类型的这些函数由于签名不兼容或者final导致而无法覆盖,那么就会报错。
-不允许为componentN()以及copy()函数提供显示实现。
在JVM中,如果生成的类需要含有一个无参的构造函数,则所有属性必须指定默认值.
data class User(val name:String=" ",val age:Int=0)
在类体中声明的属性
请注意,对于那些自动生成的函数,编译器只使用在主构造函数内部定义的属性。如需要在生成的实现中排除一个属性,请将其声明在类体中;
data class Person(val name:String){
var age:Int=0
}
在toString(),equals(),hashCode()以及copy()的实现中只会用到name属性,并且只有一个component1().虽然两个Person对象可以有不同的年龄,但他们会视为相等。
val person1=Person("John")
val person2=Person("John")
person1.age=10
person2.age=20
复制
在很多情况下,我们需要复制一个对象改变它的一些属性,但是其余部分保持不变。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)