Kotlin基础(拓展函数,委托,Object)

适合有java基础的,获取kotlin基础知识/语法等,Kotlin精讲-黑马程序员(原创)的学习笔记。

1. 拓展函数

// 拓展函数的写法
fun AnyClass.funName(params): returnType {
    // method body
}

如果kotlin里面,子类和父类拥有同名的拓展方法会是怎样呢?
==>Kotlin的拓展函数则不存在子类优先(多态)的原则。Kotlin的拓展函数是静态解析的,完全由当前变量的类型决定(不使用多态特性!)。

如果kotlin里面,类本身的成员函数和拓展函数同名会怎样呢?
==>遵循成员函数优先的原则。

2. 拓展属性

// 拓展属性的写法
val/var AnyClass.propertyName: PropertyType
getter
setter

ps: IDE小技巧,拓展函数/属性 输入 exfun, exvar, exval 会列出相应框架待填充。

3. 委托

委托模式也叫代理模式,是最常用的设计模式的一种。在委托模式中,有两个对象参与处理同一个请求,接受请求的对象将请求委托给另一个对象来处理。委托模式已证明是实现继承的一个很好的替代方式。
Kotlin 通过关键字 by 实现委托。
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的成员函数
    var luckyMoney: Int by SmallFather()
}

class SmallFather {
    var sonLuckyMoney: Int = 0

    // getValue方法,在BigSon中IDE会提示生成SmallFather的成员函数
    operator fun getValue(bigSon: BigSon, property: KProperty<*>): Int {
        println("SmallFather#getValue, ${property.name}")
        return sonLuckyMoney
    }

    // setValue方法,在BigSon中IDE会提示生成SmallFather的成员函数
    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()是一个函数, 接受一个Lambda表达式作为参数, 返回一个Lazy类型的实例,这个实例可以作为一个委托, 实现延迟加载属性(lazy property):

val normalValue = "normal value"
// lazy变量的声明(var也可以)
// 第一次调用 get() 时, 将会执行 lazy() 函数的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”的变量,我们需要保证它被正确的赋值。赋值操作可以在变量定义的时候,也可以后续代码里面进行赋值。我们只需要在变量后面使用notNull属性委托,编译器就允许我们后续进行赋值。

// 语法结构
var 变量: 变量类型 by Delegates.notNull<变量类型>()
// 或者lateinit, #lateinit不能修饰可空变量类型,且不能修饰基本类型(将变量值置null)
lateinit var 变量: 变量类型
class User {
    // property must be initialized
    // 因为我们使用了notNull的属性委托,所以编译器允许“name”后续进行赋值。
    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 实现集合类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关键字(单例,创建匿名对象,创建伴生对象)

4.1 单例

在Kotlin中,只要通过object关键字就能实现单例,简洁高效。

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 创建匿名类对象
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创建匿名类对象(实现接口,也可以继承类)
    button.setOnClickListener(object : OnClickListener {
        override fun onClick() {
            println("...#onClick...")
        }
    })
}

object声明的对象,除了实现某一个接口、继承某一个类,还可以既不实现接口,也不继承类。

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 创建伴生对象
interface ITest

open class C

class A {
    var a: String = ""
    // 伴生对象可以继承类,实现接口; 
    // 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精讲-黑马程序员(原创)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 204,445评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,889评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,047评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,760评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,745评论 5 367
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,638评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,011评论 3 398
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,669评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,923评论 1 299
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,655评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,740评论 1 330
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,406评论 4 320
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,995评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,961评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,197评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,023评论 2 350
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,483评论 2 342

推荐阅读更多精彩内容