我們經(jīng)常會(huì)創(chuàng)建一個(gè)除了持有數(shù)據(jù),而沒有其他用途的類。在這樣一個(gè)類中,一些標(biāo)準(zhǔn)的函數(shù)(setter、getter等)通常可以從數(shù)據(jù)中機(jī)械式的推導(dǎo)出來。在Kotlin中,這些類被稱為數(shù)據(jù)類,以data關(guān)鍵字標(biāo)記:
data class User(val name: String, val age: Int)
編譯器會(huì)自動(dòng)為在主構(gòu)造器中聲明的所有屬性生成以下成員:
- equals()/hashCode()函數(shù)
- "User(name=John, age=42)"的toString()函數(shù)
- 對(duì)應(yīng)于屬性的聲明順序的componentN()函數(shù)
- copy()函數(shù)
如果上述某些函數(shù)在類體中被顯式聲明,或從基類中繼承了某些函數(shù),則已存在的函數(shù)將不會(huì)生成。
為了確保生成的代碼的一致性及有效性,數(shù)據(jù)類必須滿足以下要求:
- 主構(gòu)造器至少有一個(gè)參數(shù)
- 所有構(gòu)造器參數(shù)必須以
val或var顯式標(biāo)記 - 數(shù)據(jù)類不能是抽象的,可繼承的,封閉的以及內(nèi)部的(abstract, open, sealed or inner)
- 在1.1版本之前,數(shù)據(jù)類僅能實(shí)現(xiàn)接口
從1.1之后,數(shù)據(jù)類可以繼承其他類。
在JVM上,如果生成的類需要有一個(gè)無參數(shù)的構(gòu)造函數(shù),則必須指定所有屬性的默認(rèn)值:
data class User(val name: String = "", val age: Int = 0)
復(fù)制(Copying)
通常情況下,我們需要復(fù)制一個(gè)對(duì)象,并改變其某些屬性,但是保持其他的屬性不變。這就是生成的copy()函數(shù)的作用。對(duì)于上述用戶類,其實(shí)現(xiàn)如下:
fun copy(name: String = this.name, age: Int = this.age) = User(name, age)
我們可以如此調(diào)用:
val jack = User(name = "Jack", age = 1)
val olderJack = jack.copy(age = 2)
數(shù)據(jù)類和解構(gòu)聲明(Data Classes and Destructuring Declarations)
數(shù)據(jù)類自動(dòng)生成的組件函數(shù)可用于解構(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ù)類(Standard Data Classes)
標(biāo)準(zhǔn)庫提供了Pair類和Triple類。 在大多數(shù)情況下,命名數(shù)據(jù)類則是更好的設(shè)計(jì)選擇,因?yàn)樗鼈兺ㄟ^為屬性提供有意義的名稱來使代碼更易讀。