一、創建單例
Java中構造方法私有化且使用靜態字段持有類實例來創建單例
Kotlin中的對象聲明將類聲明和單例聲明結合到了一起
object Payroll{
val allEmployees = arrayListOf<Person>()
fun calculateSalar(){
for (person in allEmployees){
//...
}
}
}
與普通類一樣,一個對象也可以包含屬性、方法、初始化語句塊等聲明,但不允許有構造方法,因爲定義時就創建了,聲明構造方法沒有任何意義
也可以繼承類和接口
//忽略大小寫比較文件路徑
object CaseInsensitiveFileComparator:Comparator<File>{
override fun compare(o1: File?, o2: File?): Int {
return o1!!.path.compareTo(o2!!.path,ignoreCase = true)
}
}
//嵌套類中的實現
data class Person(val name: String) {
object CaseInsensitiveFileComparator : Comparator<File> {
override fun compare(o1: File?, o2: File?): Int {
return o1!!.path.compareTo(o2!!.path, ignoreCase = true)
}
}
}
二、伴生對象與工廠方法
Kotlin中的類不能擁有靜態成員,作爲替代,Kotlin依賴包級別函數、對象聲明和頂層函數。
使用companion關鍵字獲得通過容器類名訪問對象和屬性的能力
class A {
companion object {
fun bar() {
}
}
}
並且伴生對象這裏是調用private構造方法的最佳選擇,伴生對象可以訪問類中的所有的private對象,更加方便工廠模式的實現
//定義一個擁有多個從構造的類
class User {
private val nickname: String
constructor(email: String) {
nickname = email.substringBefore("@")
}
constructor(facebookAccountId: Int) {
nickname = getFacebookName(facebookAccountId)
}
}
//使用工廠方法代替從構造
class User private constructor(val nickname: String) {
companion object {
fun newSubscibingUser(email: String) = User(email.substringBefore("@"))
fun newFacebookUser(facebookAccountId: Int) = User(getFacebookName(facebookAccountId))
}
}
三、伴生對象與普通對象
大多數情況通過伴生對象的類來引用伴生對象,一般不比關心他的名字,但如果需要也可以指明:
//聲明一個命名伴生對象
class Person(val name: String) {
companion object Loader {
fun fromJSON(jsonText: String): Person = ...
}
}
伴生對象的擴展
擴展函數允許定義通過代碼庫中的其他地方定義類實例的調用方法,但如果你需要定義通過類自身調用的方法:
class Person(val firstName: String, val lastName: String) {
companion object {
}
}
fun Person.Companion.fromJSON(json: String): Person {
...
}
val p = Person.fromJSON(json)
三、伴生對象與匿名內部類
//使用匿名對象來實現事件監聽器
window.addMouseListener(
object : MouseAdapter() {
override fun mouseClicked(e: MouseEvent) {
//...
}
override fun mouseEntered(e: MouseEvent) {
//...
}
}
)
//或者
val listener = object : MouseAdapter() {
override fun mouseClicked(e: MouseEvent) {
//...
}
override fun mouseEntered(e: MouseEvent) {
//...
}
}
匿名對象也可以訪問局部變量
fun countClicks(window: Window) {
val clickCount = o
window.addMouseListener(
object : MouseAdapter() {
override fun mouseClicked(e: MouseEvent) {
clickCount++
}
}
)
}