在开发过程中,经常会创建一些数据里,其没有任何逻辑功能,仅仅来用来保存数据。在Kolin中,将这些类统一称为数据类,用关键字data标记。
data class User(val name: String, val age: Int)
编译器会根据主构造器中声明的全部属性, 自动推断产生以下成员函数:
如果在该数据类或者基类中重写了以上某个成员函数,将不会再自动推断,以重写的为准。
如果数据类有无参构造函数,需将主构造函数中,将成员属性初始化。
data class User(val name: String = "", val age: Int = 0)
之前提到,JVM会自动为一个数据类生成copy()函数。copy()到底用来干啥的?copy()是对对象的浅拷贝还是深拷贝?带着这些疑问往下看。
data class Person(var name: String, var age: Int) {
}
data class MyInfo(val no: Int, var person: Person) {
override fun hashCode(): Int {
return super.hashCode()
}
}
fun main(args: Array) {
var per0: Person = Person("A", 20)
var per1: Person = per0.copy(age = 15)
println("------- Log0 -------")
println("per0: $per0")
println("per1: $per1")
println("------- Log1 -------")
per1.name = "B"
println("per0: $per0")
println("per1: $per1")
println("------- Log2 -------")
var info0: MyInfo = MyInfo(1, per0)
var perAA = Person("AA", 30)
var info1: MyInfo = info0.copy(no = 2)
var info2: MyInfo = info0.copy(person = perAA)
println("info0: $info0")
println("info1: $info1")
println("------- Log3 -------")
info1.person.name = "C"
println("info0: $info0")
println("info1: $info1")
println("------- Log4 -------")
println("info0: $info0")
println("info1: $info1")
println("info2: $info2")
println("------- Log5 -------")
info2.person.name = "AC"
println("info0: $info0")
println("info1: $info1")
println("info2: $info2")
println("perAA: $perAA")
}
// Log打印
------- Log0 -------
per0: Person(name=A, age=20)
per1: Person(name=A, age=15)
------- Log1 -------
per0: Person(name=A, age=20)
per1: Person(name=B, age=15)
------- Log2 -------
info0: MyInfo(no=1, person=Person(name=A, age=20))
info1: MyInfo(no=2, person=Person(name=A, age=20))
------- Log3 -------
info0: MyInfo(no=1, person=Person(name=C, age=20))
info1: MyInfo(no=2, person=Person(name=C, age=20))
------- Log4 -------
info0: MyInfo(no=1, person=Person(name=C, age=20))
info1: MyInfo(no=2, person=Person(name=C, age=20))
info2: MyInfo(no=1, person=Person(name=AA, age=30))
------- Log5 -------
info0: MyInfo(no=1, person=Person(name=C, age=20))
info1: MyInfo(no=2, person=Person(name=C, age=20))
info2: MyInfo(no=1, person=Person(name=AC, age=30))
perAA: Person(name=AC, age=30)
先创建了两个数据类Person和MyInfo,而MyInfo类中有一个Person成员属性,现分别创建了Person和MyInfo实例,并分别运用了其copy()函数,生成了一个新的对象。
从Log0和Log1可以看出,通过copy()得到的对象修改其基本类型的属性时,对原对象并没有影响。
从Log2和Log3的对比及Log4和Log5的对比,在执行copy()函数时,对于引用类型,其引用对象将指向被复制过的新对象,并不是新创建了一个对象。
也就是说,copy()函数实现的功能是复制一个对象,然后修改其一部分属性,但保持其他属性保持不变。copy并不是简单的复制,它会创建新的对象,对于基本类型的属性,其值会被拷贝并赋值,但是对于引用类型,其只是将其地址传递过去,还是进行浅拷贝,并没有创建新的对象。怎么觉得copy()实际上是浅拷贝和深拷贝的综合呢,好不理解哦。
对于“=”号赋值,大家都很清楚的是进行了浅拷贝,从Log1中,就可以即使基本数据类型,在新的对象的基本类型属性进行赋值时,也会影像原值的属性。
var per0: Person = Person("A", 20)
var per1: Person = per0
------- Log0 -------
per0: Person(name=A, age=20)
per1: Person(name=A, age=20)
------- Log1 -------
per0: Person(name=B, age=20)
per1: Person(name=B, age=20)
“=”号赋值和copy()在执行机制上还是有区别的。在对copy使用时,还是要注意对引用类型的浅拷贝,稍微留神就犯大错误了。。。
前面提到了,JVM会数据类生成一些列的组件函数(componentN() 函数群),其与这些函数与类的属性对应, 函数名中的数字 1 到 N, 与属性的声明顺序一致,有了这些组件函数, 就可以在解构声明中使用数据。
val jane = Person("Jane", 35)
val (n, a) = jane
println("$n, $a years of age") // 打印结果将是 "Jane, 35 years of age"