Kotlin基礎(拓展函數(shù)屉来,委托,Object)

適合有java基礎的狈癞,獲取kotlin基礎知識/語法等茄靠,Kotlin精講-黑馬程序員(原創(chuàng))的學習筆記。

1. 拓展函數(shù)

// 拓展函數(shù)的寫法
fun AnyClass.funName(params): returnType {
    // method body
}

如果kotlin里面蝶桶,子類和父類擁有同名的拓展方法會是怎樣呢慨绳?
==>Kotlin的拓展函數(shù)則不存在子類優(yōu)先(多態(tài))的原則。Kotlin的拓展函數(shù)是靜態(tài)解析的真竖,完全由當前變量的類型決定(不使用多態(tài)特性F暄)。

如果kotlin里面恢共,類本身的成員函數(shù)和拓展函數(shù)同名會怎樣呢战秋?
==>遵循成員函數(shù)優(yōu)先的原則。

2. 拓展屬性

// 拓展屬性的寫法
val/var AnyClass.propertyName: PropertyType
getter
setter

ps: IDE小技巧讨韭,拓展函數(shù)/屬性 輸入 exfun, exvar, exval 會列出相應框架待填充脂信。

3. 委托

委托模式也叫代理模式,是最常用的設計模式的一種透硝。在委托模式中狰闪,有兩個對象參與處理同一個請求,接受請求的對象將請求委托給另一個對象來處理濒生。委托模式已證明是實現(xiàn)繼承的一個很好的替代方式埋泵。
Kotlin 通過關鍵字 by 實現(xiàn)委托。
Kotlin中委托分為 (委托類) 和 (委托屬性),Kotlin官方庫也封裝了一些常用的委托丽声。

3.1 委托類
// (委托類)
interface IWash {
    fun wash()
}

class BigSon : IWash {
    override fun wash() {
        println("bigSon wash...")
    }
}

// 代理模式/委托模式 (委托類)
class SmallFather(private val iWash: IWash): IWash by iWash

fun main(args: Array<String>) {
    val bigSon = BigSon()
    val smallFather = SmallFather(bigSon)
    // 圍裙媽媽讓小頭爸爸洗碗
    println("叮囑大頭兒子洗碗認真點")
    smallFather.wash()
    println("檢查大頭兒子洗碗是否干凈")
}
3.2 委托屬性

val/var propertyName: propertyType by <expression>

class BigSon {
    // IDE會提示生成SmallFather的成員函數(shù)
    var luckyMoney: Int by SmallFather()
}

class SmallFather {
    var sonLuckyMoney: Int = 0

    // getValue方法礁蔗,在BigSon中IDE會提示生成SmallFather的成員函數(shù)
    operator fun getValue(bigSon: BigSon, property: KProperty<*>): Int {
        println("SmallFather#getValue, ${property.name}")
        return sonLuckyMoney
    }

    // setValue方法,在BigSon中IDE會提示生成SmallFather的成員函數(shù)
    operator fun setValue(bigSon: BigSon, property: KProperty<*>, value: Int) {
        println("SmallFather#setValue, ${property.name} => $value")
        sonLuckyMoney = value
    }
}

fun main(args: Array<String>) {
    val bigSon = BigSon()
    bigSon.luckyMoney = 100
    bigSon.luckyMoney -= 50

    println("son's luckyMoney is ${bigSon.luckyMoney}")
}

==>
SmallFather#setValue, luckyMoney => 100
SmallFather#getValue, luckyMoney
SmallFather#setValue, luckyMoney => 50
SmallFather#getValue, luckyMoney
son's luckyMoney is 50
3.3 五大內置委托
3.3.1 延遲加載(Lazy)

lazy()是一個函數(shù), 接受一個Lambda表達式作為參數(shù), 返回一個Lazy類型的實例,這個實例可以作為一個委托, 實現(xiàn)延遲加載屬性(lazy property):

val normalValue = "normal value"
// lazy變量的聲明(var也可以)
// 第一次調用 get() 時, 將會執(zhí)行 lazy() 函數(shù)的Lambda 表達式雁社,以后調用直接返回結果(最后一行)
val lazyValue: String by lazy {
    println("init lazyValue only once")
    "Hello lazy"
}

fun main(args: Array<String>) {
    println(normalValue)
    println(lazyValue)
    println(lazyValue)
}
3.3.2可觀察屬性(Observable)
// 語法結構
var 變量: 變量類型 by Delegates.observable(變量初始化值) {
    propety, oldValue, newValue ->
        doSth..(可以多行)
}
3.3.3 Vetoable(veto:否決)
// 語法結構
var 變量: 變量類型 by Delegates.vetoable(變量初始化值) {
    propety, oldValue, newValue ->
        doSth..(可以多行)
        // true瘦麸,變量會被修改;false歧胁,變量不會被修改
        true/false 
}
3.3.4 notNull

對于一個不可為“non-null”的變量,我們需要保證它被正確的賦值厉碟。賦值操作可以在變量定義的時候喊巍,也可以后續(xù)代碼里面進行賦值。我們只需要在變量后面使用notNull屬性委托箍鼓,編譯器就允許我們后續(xù)進行賦值崭参。

// 語法結構
var 變量: 變量類型 by Delegates.notNull<變量類型>()
// 或者lateinit, #lateinit不能修飾可空變量類型款咖,且不能修飾基本類型(將變量值置null)
lateinit var 變量: 變量類型
class User {
    // property must be initialized
    // 因為我們使用了notNull的屬性委托何暮,所以編譯器允許“name”后續(xù)進行賦值。
    var name: String by notNull<String>()
}

fun main(args: Array<String>) {
    val user = User()
    // user.name并沒有進行賦值铐殃,運行時會報錯海洼!
    // Exception in thread "main" java.lang.IllegalStateException: Property name should be initialized before get.
    println(user.name)
}
3.3.5 將多個屬性保存在一個map內

Kotlin的map委托,提供了map和對象一一綁定關系富腊,就是map的值可以決定對象的值坏逢,修改map的值也可以影響對象的值。但是這一切需要滿足赘被,map的key和屬性的名稱保證一致是整。

// 語法結構
val/var 變量: 變量類型 by 實現(xiàn)集合類Map接口的對象
class User(map: MutableMap<String, Any>) {
    var name: String by map
    var age: Int by map
}

fun main(args: Array<String>) {
    val map = mutableMapOf(
            "name" to "jerry",
            "age" to 4
    )
    val user = User(map)
    println(user.name)
    map["name"] = "Jerry"
    println(user.name)
    println(user.age)
}

4. Object關鍵字(單例,創(chuàng)建匿名對象民假,創(chuàng)建伴生對象)

4.1 單例

在Kotlin中浮入,只要通過object關鍵字就能實現(xiàn)單例,簡潔高效羊异。

object Dog {
    var name: String = ""
    var age: Int = 1
}

fun main(args: Array<String>) {
    // ^^^不是 Dog()^^^
    val dog1 = Dog
    val dog2 = Dog
    // dog1, dog2 同一個對象
    println(dog1)
    println(dog2)
}
4.2 創(chuàng)建匿名類對象
interface OnClickListener {
    fun onClick()
}

class ImageButton {
    private var listener: OnClickListener? = null
    fun setOnClickListener(listener: OnClickListener) {
        this.listener = listener
    }
}

fun main(args: Array<String>) {
    val button = ImageButton()
    // object創(chuàng)建匿名類對象(實現(xiàn)接口事秀,也可以繼承類)
    button.setOnClickListener(object : OnClickListener {
        override fun onClick() {
            println("...#onClick...")
        }
    })
}

object聲明的對象,除了實現(xiàn)某一個接口球化、繼承某一個類秽晚,還可以既不實現(xiàn)接口,也不繼承類筒愚。

fun main(args: Array<String>) {
    // 直接聲明一個匿名類
    val person = object {
        var name: String = "Jerry"
        var age: Int = 4
    }
    println("person's name is ${person.name}")
    println("person's age is ${person.age}")
}
4.3 創(chuàng)建伴生對象
interface ITest

open class C

class A {
    var a: String = ""
    // 伴生對象可以繼承類赴蝇,實現(xiàn)接口; 
    // B 可以省略
    companion object B: C(), ITest {
        var b: String = "Jerry"
        fun bFun() {
            println("..#bFun()..")
        }
    }
}

fun main(args: Array<String>) {
    // 訪問伴生對象的屬性的2種方式
    println(A.b)
    println(A.B.b)
    // 訪問伴生對象的方法的2種方式
    println(A.bFun())
    println(A.B.bFun())
}

參考:
Kotlin精講-黑馬程序員(原創(chuàng))

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯(lián)系作者
  • 序言:七十年代末巢掺,一起剝皮案震驚了整個濱河市句伶,隨后出現(xiàn)的幾起案子劲蜻,更是在濱河造成了極大的恐慌,老刑警劉巖考余,帶你破解...
    沈念sama閱讀 219,110評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件先嬉,死亡現(xiàn)場離奇詭異,居然都是意外死亡楚堤,警方通過查閱死者的電腦和手機疫蔓,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,443評論 3 395
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來身冬,“玉大人衅胀,你說我怎么就攤上這事∷煮荩” “怎么了滚躯?”我有些...
    開封第一講書人閱讀 165,474評論 0 356
  • 文/不壞的土叔 我叫張陵,是天一觀的道長嘿歌。 經(jīng)常有香客問我掸掏,道長,這世上最難降的妖魔是什么宙帝? 我笑而不...
    開封第一講書人閱讀 58,881評論 1 295
  • 正文 為了忘掉前任丧凤,我火速辦了婚禮,結果婚禮上茄唐,老公的妹妹穿的比我還像新娘息裸。我一直安慰自己,他們只是感情好沪编,可當我...
    茶點故事閱讀 67,902評論 6 392
  • 文/花漫 我一把揭開白布呼盆。 她就那樣靜靜地躺著,像睡著了一般蚁廓。 火紅的嫁衣襯著肌膚如雪访圃。 梳的紋絲不亂的頭發(fā)上,一...
    開封第一講書人閱讀 51,698評論 1 305
  • 那天相嵌,我揣著相機與錄音腿时,去河邊找鬼。 笑死饭宾,一個胖子當著我的面吹牛批糟,可吹牛的內容都是我干的。 我是一名探鬼主播看铆,決...
    沈念sama閱讀 40,418評論 3 419
  • 文/蒼蘭香墨 我猛地睜開眼徽鼎,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起否淤,我...
    開封第一講書人閱讀 39,332評論 0 276
  • 序言:老撾萬榮一對情侶失蹤悄但,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后石抡,有當?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體檐嚣,經(jīng)...
    沈念sama閱讀 45,796評論 1 316
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 37,968評論 3 337
  • 正文 我和宋清朗相戀三年啰扛,在試婚紗的時候發(fā)現(xiàn)自己被綠了嚎京。 大學時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 40,110評論 1 351
  • 序言:一個原本活蹦亂跳的男人離奇死亡隐解,死狀恐怖挖藏,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情厢漩,我是刑警寧澤,帶...
    沈念sama閱讀 35,792評論 5 346
  • 正文 年R本政府宣布岩臣,位于F島的核電站溜嗜,受9級特大地震影響,放射性物質發(fā)生泄漏架谎。R本人自食惡果不足惜炸宵,卻給世界環(huán)境...
    茶點故事閱讀 41,455評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望谷扣。 院中可真熱鬧土全,春花似錦、人聲如沸会涎。這莊子的主人今日做“春日...
    開封第一講書人閱讀 32,003評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽末秃。三九已至概页,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間练慕,已是汗流浹背惰匙。 一陣腳步聲響...
    開封第一講書人閱讀 33,130評論 1 272
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留铃将,地道東北人项鬼。 一個月前我還...
    沈念sama閱讀 48,348評論 3 373
  • 正文 我出身青樓,卻偏偏與公主長得像劲阎,于是被迫代替她去往敵國和親绘盟。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 45,047評論 2 355

推薦閱讀更多精彩內容