數(shù)據(jù)類
Kotlin 可以創(chuàng)建一個(gè)只包含數(shù)據(jù)的類固蛾,關(guān)鍵字為 data:
data class User(val name: String, val age: Int)
編譯器會(huì)自動(dòng)的從主構(gòu)造函數(shù)中根據(jù)所有聲明的屬性提取以下函數(shù):
equals() / hashCode()
toString() 格式如 "User(name=John, age=42)"
componentN() functions 對(duì)應(yīng)于屬性扒披,按聲明順序排列
copy() 函數(shù)
如果這些函數(shù)在類中已經(jīng)被明確定義了饥侵,或者從超類中繼承而來,就不再會(huì)生成。
為了保證生成代碼的一致性以及有意義萄凤,數(shù)據(jù)類需要滿足以下條件:
主構(gòu)造函數(shù)至少包含一個(gè)參數(shù)。
所有的主構(gòu)造函數(shù)的參數(shù)必須標(biāo)識(shí)為val 或者 var ;
數(shù)據(jù)類不可以聲明為 abstract, open, sealed 或者 inner;
數(shù)據(jù)類不能繼承其他類 (但是可以實(shí)現(xiàn)接口)搪哪。
復(fù)制
復(fù)制使用 copy() 函數(shù)靡努,我們可以使用該函數(shù)復(fù)制對(duì)象并修改部分屬性, 對(duì)于上文的 User 類,其實(shí)現(xiàn)會(huì)類似下面這樣:
fun copy(name: String = this.name, age: Int = this.age) = User(name, age)
使用 copy 類復(fù)制 User 數(shù)據(jù)類,并修改 age 屬性:
data class User(val name: String, val age: Int)
fun main(args: Array<String>) {
val jack = User(name = "Jack", age = 1)
val olderJack = jack.copy(age = 2)
println(jack)
println(olderJack)
}
數(shù)據(jù)類以及解構(gòu)聲明
組件函數(shù)允許數(shù)據(jù)類在解構(gòu)聲明中使用:
val jane = User("Jane", 35)
val (name, age) = jane
println("$name, $age years of age") // prints "Jane, 35 years of age"
標(biāo)準(zhǔn)數(shù)據(jù)類
標(biāo)準(zhǔn)庫提供了 Pair 和 Triple 惑朦。在大多數(shù)情形中兽泄,命名數(shù)據(jù)類是更好的設(shè)計(jì)選擇,因?yàn)檫@樣代碼可讀性更強(qiáng)而且提供了有意義的名字和屬性漾月。
密封類
密封類用來表示受限的類繼承結(jié)構(gòu):當(dāng)一個(gè)值為有限幾種的類型, 而不能有任何其他類型時(shí)病梢。在某種意義上,他們是枚舉類的擴(kuò)展:枚舉類型的值集合 也是受限的梁肿,但每個(gè)枚舉常量只存在一個(gè)實(shí)例蜓陌,而密封類 的一個(gè)子類可以有可包含狀態(tài)的多個(gè)實(shí)例。
聲明一個(gè)密封類吩蔑,使用 sealed 修飾類钮热,密封類可以有子類,但是所有的子類都必須要內(nèi)嵌在密封類中哥纫。
sealed 不能修飾 interface ,abstract class(會(huì)報(bào) warning,但是不會(huì)出現(xiàn)編譯錯(cuò)誤)
sealed class Expr
data class Const(val number: Double) : Expr()
data class Sum(val e1: Expr, val e2: Expr) : Expr()
object NotANumber : Expr()
fun eval(expr: Expr): Double = when (expr) {
is Const -> expr.number
is Sum -> eval(expr.e1) + eval(expr.e2)
NotANumber -> Double.NaN
}
使用密封類的關(guān)鍵好處在于使用when表達(dá)式的時(shí)候霉旗,如果能夠驗(yàn)證語句覆蓋了所有情況,就不需要為該語句再添加一個(gè) else 子句了蛀骇。上面的類都繼承了密封類Expr 所以Expr也就限定在了這些類型里面厌秒。使用密封類的關(guān)鍵好處在于使用 when 表達(dá)式 的時(shí)候,如果能夠 驗(yàn)證語句覆蓋了所有情況擅憔,就不需要為該語句再添加一個(gè) else 子句了鸵闪。
fun eval(expr: Expr): Double = when(expr) {
is Expr.Const -> expr.number
is Expr.Sum -> eval(expr.e1) + eval(expr.e2)
Expr.NotANumber -> Double.NaN
// 不再需要 `else` 子句,因?yàn)槲覀円呀?jīng)覆蓋了所有的情況
}