Kotlin边用边学:别把Extension Function玩坏了

Key Takeaways(划重点)

  • 设计原则不要忘
  • 扩展是个补锅匠
  • 定向输出是好手
  • 升级外挂要当心

这里假定你对Kotlin的Extension Functioins(扩展函数)有一定了解,如果没有,请先浏览官网有关它的定义和常见用法。

面向对象编程

在Kotlin的官方网站介绍扩展函数的时候,使用的例子是Collections.swap(List, int, int),那我们也拿它来说事吧。

这个是swap在Java中的定义:

Collections.java

public class Collections {
    /** @since 1.4 */
    public static void swap(List<?> list, int i, int j)
}

看到这个,我就琢磨一个事,为什么我不能直接把这个方法定义在List上呢,好比这样:

List.java

public interface List<E> extends Collection<E> {
    void swap(int i, int j)
}

如果这样可行,哪里还要Collections.swap啊。

但现实是这个Collections.swap()是从Java 1.4就有的,而想要List能够原生支持swap得等到Java 8的Interface Default Methods

public interface List<E> extends Collection<E> {
    void swap(int i, int j) {
        set(i, l.set(j, get(i))); // 不是Java 8+编译报错
    }
}

所以,之前我们认为的别扭设计,不是Java的设计师没想到,而真是时机未到。

那如果我们在Kotlin中也来这么玩一下(当然是用扩展函数而不是工具类了):

fun MutableList<E>.swap(i: int, j: int): Unit

从使用的角度,和原生函数是一样的:mutableListOf(1, 2, 3).swap(0, 2)。但这个时候我们就需要问一句,为什么不直接使用原生的实现呢?

所以,基于设计原则的原生函数是始终优先于扩展函数的。(千万别忘了OO的多态的强大,而扩展函数恰恰由于是静态调用,不支持多态的)

扩展函数是个补锅匠

Kotlin自身就有很多扩展函数,譬如Array<T>.map(transform: (T) -> R): List<R>。这个扩展可以让数组按需转换,也是一个高频使用的扩展函数。

public inline fun <T, R> Array<out T>.map(transform: (T) -> R): List<R> {
    return mapTo(ArrayList<R>(size), transform)
}

看到这里可能就有疑问,为什么不做成Array的原生函数呢?

其实,如果阅读代码够仔细,可以看到,接口定义和实现中分别出现了List<R>ArrayList<R>,这些代表着依赖。Array本身其实是不应该依赖他们的,如果把这个map功能进行原生实现,就存在接口污染问题(高内聚低耦合设计原则)。

所以这个时候,扩展函数就可以很好的补上这个锅 了。

特定场景的定向扩展

谈到原生函数,一般而言只会存放普遍需要的。对于只适用于某些特定场合/环境/上下文的,就不适合了。譬如Domain Object或OOA的Entity,如Person,在某些场景需要输出成json的时候,如果可以直接person.toJson()会很便捷。但这个不是基础需求,也不希望引入JSON带来第三方类/库。另外这个口子开了,后面其他使用者需要输出成xml的时候,同样会再次引入xml而带来更多的第三方类/库。

这样原本的一个存粹的Domain Object或Entity或者说我们的common/core module,依赖了一堆堆第三方包,咋看都怪怪的。

这个时候,扩展函数又是一个很好的输出。在需要的应用中,我们可以自行扩展:

// app: JSON app
fun Person.toJson(): String
// app: XML app
fun Person.toXml(): String

原生函数执行优先级高于扩展函数

对第三方库进行扩展的时候,需要注意扩展函数和原生函数在执行上的一个先后顺序:原生函数优先于扩展函数。

If a class has a member function, and an extension function is defined which has the same receiver type, the same name and is applicable to given arguments, the member always wins

这个问题在升级被扩展的类库的时候尤其需要注意。拿上面例子中的toJson来说,本来原生没提供,所以我们扩展了。但对方可能也意识到这个蛮有用(甭纠结上面不是说违反原则吗),也就在某次版本中添加了这个功能。问题是对方的原生实现在null值处理的时候和你不一样(假如对方null的时候String会是空串 "name": "",而你是不输出),就可能导致升级之后由于原生函数优先执行,你的程序可能就会挂了。

所以,玩外挂每次升级都需要额外注意

总结下扩展函数的使用要点/场景:

  • 设计原则不要忘
  • 扩展是个补锅匠
  • 定向输出是好手
  • 升级外挂要当心

希望这篇博文能对你有所帮助,喜欢的话点个赞吧👍!

更多Kotlin的实用技巧,请参考《Kotlin边用边学系列

参考资料

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

推荐阅读更多精彩内容