【kotlin】委托

在 kotlin 开发中,会遇到懒加载的情形:使用 by lazy 关键字。而这是通过委托来实现的。Kotlin 通过关键字 by 实现委托。

委托分为类委托和属性委托。

一、类委托

通过类委托,一个对象可以将继承自某个接口的函数全部交由另一个对象来实现。其基本形式为:

    interface Interface {
        fun function1(params: String)
        fun function2()
    }

    class Impl(delegate: Interface) : Interface by delegate

    class Delegate : Interface {
        override fun function1(params: String) {
            println("delegate function1 $params")
        }

        override fun function2() {
            println("delegate function2")
        }
    }

在以上代码中,Impl 类的所有继承自 Interface 接口的函数都交由 delegate 对象来实现。

类委托的例子

类委托不仅仅只局限于单一接口。例如,假设现有一个抽象类 PhoneManufacturer(手机生产商),定义如下:

    open class PhoneManufacturer(val name: String) : ScreenManufacturer, CpuManufacturer, Assembler

    interface ScreenManufacturer {
        fun produceScreen()
    }

    interface CpuManufacturer {
        fun produceCpu()
    }

    interface Assembler {
        fun assemble()
    }

其中,手机生产商 PhoneManufacturer 继承了三个接口,分别为 ScreenManufacturer(屏幕生产商)、CpuManufacturer(Cpu生产商)、Assembler(组装厂)。
但是,由于手机生产商能力有限,无法将所有环节全部由自己制造,所以需要将一部分环节委托给代工厂。于是,利用委托模式,可以将 PhoneManufacturer 修改为以下形式:

    open class PhoneManufacturer(
        val name: String,
        private val screenManufacturer: ScreenManufacturer,
        private val cpuManufacturer: CpuManufacturer,
        private val assembler: Assembler
    ) : ScreenManufacturer by screenManufacturer,
        CpuManufacturer by cpuManufacturer,
        Assembler by assembler {
        fun sell() {
            println("${name}卖出了一台由${screenManufacturer}生产屏幕、${cpuManufacturer}生产Cpu、${assembler}组装的手机")
        }
    }

于是,手机生产商就可以将屏幕生产、Cpu生产、组装全部外包出去,只管销售就行了。
现在,我们来测试一下效果。

    fun main(args: Array<String>) {
        val xiaomi = PhoneManufacturer("Xiaomi", Samsung(), Qualcomm(), Foxconn())
        xiaomi.sell()
    }
    class Samsung : ScreenManufacturer {
        override fun produceScreen() = Unit
        override fun toString(): String = "Samsung"
    }
    class Qualcomm : CpuManufacturer {
        override fun produceCpu() = Unit
        override fun toString(): String = "Qualcomm"
    }
    class Foxconn : Assembler {
        override fun assemble() = Unit
        override fun toString() = "Foxconn"
    }

在这里,我们新建了三个类:三星作为屏幕生产商、高通作为Cpu生产商、富士康作为组装厂;然后创建手机厂商对象 xiaomi,并调用其 sell 函数。

输出:

Xiaomi卖出了一台由Samsung生产屏幕、Samsung生产Cpu、Foxconn组装的手机

可以看到,委托成功。也就是说,PhoneManufacturer 类虽然实现了上述三个接口,但是并没有自己实现其中的函数,而是委托给其他对象来实现了。这就是类委托。

二、属性委托

在开发中经常会遇到需要懒加载的情况:某个值在类创建的时候无法确定,需要在使用的时候进行处理。这个时候,通常可以使用 by lazy 进行数据的懒加载。

这有点类似于 lateinit var,不过是在第一次调用的时候进行数据加载。

例如,在常规 Android 开发中,控件通常定义为 lateinit var 类型,并在 onCreate 回调之中进行 findViewById 查找控件的。

    private lateinit var image: ImageView

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_main)

        image = findViewById(R.id.img)
    }

而使用 by lazy 进行懒加载,也是一种方式;它可以将控件的查找延后到第一次使用的时候。这样可以减少 onCreate 中要做的事,并且使代码看起来更简洁。

    private val image: ImageView by lazy { findViewById(R.id.img) }

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_crop_bmp)
    }

而懒加载利用到的就是 kotlin 中的属性委托。

如果想自己实现一个委托,很简单,只用创建一个类,并且在其中添加一个 getValue 函数返回需要的值即可。例如,我想要实现一个 findViewById 的委托,只需要添加这么一个扩展函数和委托类即可:

    fun <T : View> Activity.findView(id: Int) = FindViewDelegate<T>(id)

    class FindViewDelegate<T : View>(val id: Int) {
        operator fun getValue(activity: Activity, kProperty: KProperty<*>): T {
            return activity.findViewById(id)
        }
    }

其中,getValue 的第一个参数必须是属性所在类的类型,在这里是 Activity;第二个参数是 KProperty<*> 类型,对应了需要被代理的那个属性。

然后,在任意的 Activity 中,均可以使用 findView 扩展函数进行属性委托了。

class MainActivity : AppCompatActivity() {
    private val image: ImageView by findView(R.id.img)

    override fun onCreate(savedInstanceState: Bundle?) {
        super.onCreate(savedInstanceState)
        setContentView(R.layout.activity_crop_bmp)
    }
}

其他系统提供的属性委托

kotlin.properties.Delegates 工具类提供了一些系统自带的属性委托。

1. notNull

委托于 Delegates.notNull 的变量类似于 lateinit var,只是一个类似于占位符的东西,表示这个对象不能为空,作用只是通过编译。在实际使用之前,还是需要将其进行初始化。

    var name: String by Delegates.notNull()

    fun main(args: Array<String>) {
        val nnd = NotNullDelegate()
        nnd.name = "Bob"
        println(nnd.name)
    }

2. observable

Delegates.observable 用于实现观察者模式,监听变量的修改。

    var name: String by Delegates.observable("Bob") { property, oldValue, newValue ->
        notifyNameChanged(property, oldValue, newValue)
    }

    fun notifyNameChanged(property: KProperty<*>, oldValue: String, newValue: String) {
        // do something
    }

3. vetoable

Delegates.vetoable 同样可以监听变量的修改,并且能够拦截这次修改。如果修改被拦截,则变量的值保持不变。

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