Kotlin(七)、object關鍵字

一、創建單例

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++
                }
            }
    )
}
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章