有时候把一个对象解构成很多变量会很方便,例如:
class person(val name:String,val age:Int){
operator fun component1()= name
operator fun component2()= age
}
val(name,age) = person
这种语法成为解构声明,一个解构声明可以同时闯进多个变量,比如上面的例子,就声明了两个新变量:name和age,并且可以独立使用它们。
但是需要先为字段声明componentN函数,并且用operator修饰它们。
println(name)
println(age)
一个解构声明会被编译成以下代码:
val name = person.component1()
val age = person.component2()
其中的 component1() 和 component2() 函数是在 Kotlin 中广泛使用的约定原则。
任何表达式都可以出现在解构声明的右侧,只要可以对它调用所需数量的 component 函数即可。 当然,可以有 component3() 和 component4() 等等。
解构声明也可以用在for循环中:
for((a,b) in collection){....}
变量 a 和 b 的值取自对集合中的元素上调用 component1() 和 component2() 的返回值。
举个场景:
让我们假设我们需要从一个函数返回两个东西。
例如,一个结果对象和一个某种状态。 在 Kotlin 中一个简洁的实现方式是声明一个数据类 并返回其实例:
data class Result(val result: Int, val status: Status)
fun function(……): Result {
// 各种计算
return Result(result, status)
}
// 现在,使用该函数:
val (result, status) = function(……)
因为数据类自动声明 componentN() 函数,所以这里可以用解构声明。
或者我们可以看看Kotlin是怎么巧用解耦声明的:
在Kotlin中,遍历一个Map最好的方式是这样
for((key,value) in map){
do something....
}
那Kotlin是如何拓展Map的?它使用到我们的扩展函数与扩展对象
operator fun <K, V> Map<K, V>.iterator(): Iterator<Map.Entry<K, V>> = entrySet().iterator()
operator fun <K, V> Map.Entry<K, V>.component1() = getKey()
operator fun <K, V> Map.Entry<K, V>.component2() = getValue()
我们可以看出
很巧妙的使用拓展属性为Entry对象拓展解构函数必须声明的componentN函数,简介的为Entry支持了解耦特性。