帮助理解Kotlin函数的3个概念

kotlin&android

人生苦短,我选Kotlin
——笔者

Kotlin相比Java很年轻,也更有潜力,其对函数式编程的支持也让其代码更加简洁,但在理解函数式编程的过程中,总会有些障碍,比如笔者在看到apply和with这两个方法的时候,就很奇怪,看了源码就更加糊涂了,本文以剖析apply和with这两个方法为线索,介绍下kotlin中函数式编程相关的几个概念。

函数类型(function type)

在函数式语言中,函数作为一种类型可以在函数间传递,那么如何区别不同的函数的类型呢?
将函数的入参和返回值,作为一种函数类型,比如:
(Int) -> Int 是一个函数类型,它的传入参数为Int,返回类型为Int,满足这个条件的函数为同一种类型的函数。

fun double(x: Int): Int {
    return 2 * x
}

比如double这个函数的类型为(Int) -> Int,它的入参是Int,返回值为Int.

由于函数为一种类型,我们可以像定义Int值一样定义一个函数类型的变量:

val double: (Int) -> Int = fun(value: Int): Int {
    return 2 * value
}

double的类型为函数,函数类型为(Int) -> Int
函数作为变量可以传递:

val doubleCopy = double
doubleCopy(1)

此时doubleCopy的类型和double的类型相同

高阶函数 (high order function)

如果一个函数将一个函数作为参数,或者返回一个函数,那么这种函数叫做high order function(高阶函数)

前面提到函数可以作为变量进行传递,将函数作为参数传递到另一个函数中,也是允许的,比如下面的例子:

// lock为高阶函数,接受2个参数,类型分别为:Lock,() -> T
// 前者为常见的Lock类型,后者为一个函数类型,这个类型的函数入参为空,返回值为T
fun <T> lock(lock: Lock, body: () -> T): T {
    lock.lock()
    try {
        return body()
    }
    finally {
        lock.unlock()
    }
}
// 调用方法
lock (lock,{sharedResource.operation()})
// 在kotlin中,如果一个函数的最后一个参数为函数,则可以将这个函数的函数体放到括号外面,就是常见的下面这种写法
lock (lock) {
    sharedResource.operation()
}

function-with-receiver type

kotlin中存在一个比较特殊的函数类型定义,它指定了函数的receiver(看起来和扩展方法比较像)

理解这个很关键,kotlin中很多基本的函数都是基于这种函数类型来实现的。

比如下面的代码段中,声明了intToLong这个函数的receiver类型为Int,只有Int类型的对象(A)可以调用intToLong这个方法,并且在intToLong函数体内,可以通过this来调用A中的函数

val intToLong: Int.() -> Long = { toLong() }

上面的代码段中,实际上调用的是Int类型本身定义的toLong方法:

//this可以省略掉
val intToLong: Int.() -> Long = { this.toLong() }
//可以编译通过,调用时,上面一句中的this即为3
val Long a = 3.intToLong()
//不可以编译通过,intToLong声明了receiver,只能被Int类型调用
val Long b = "3".intToLong()

分析下appy和with方法

这两个方法是kotlin中常用的方法,可以简化代码,让逻辑更加清晰整洁,但直接理解起来会有点绕,如果你看懂了前面讲到的几个概念之后,appy和with方法理解起来就相对容易很多

apply

apply是kotlin中常用的方法,它的官方文档中的定义是这样的:

apply:Calls the specified function block with this value as its receiver and returns this value.

我们来看下他的实现代码:

public inline fun <T> T.apply(block: T.() -> Unit): T { block(); return this }

这里结合前面的函数相关概念,对这个方法进行分析:

apply为高阶函数,它接受一个参数block,类型为 T.() -> Unit,在apply的函数体内,调用了传入的block这个函数,然后返回调用apply函数的对象实例。

需要注意的是,block函数的类型为 function-with-receiver ,在block函数体内,可以通过this访问到T类型的实例。

//调用方法
fun getDeveloper(): Developer {
    return Developer().apply {
        developerName = "Amit Shekhar"
        developerAge = 22
    }
}
// 等同于下面这个方法
fun getDeveloper(): Developer {
    //apply 方法返回新创建的Developer()
    return Developer().apply {
        //this 为新创建的Developer(),可省略
        this.developerName = "Amit Shekhar"
        this.developerAge = 22
    }
}

with

with也是比较常用的方法,它的定义是这样的:

Calls the specified function block with the given receiver as its receiver and returns its result.

它的实现代码也只有一行

public inline fun <T, R> with(receiver: T, block: T.() -> R): R = receiver.block()

with为高阶函数,接收两个参数:receiver,类型为T,block 类型为 T.() -> R,为function-with-receiver type,只能被T类型的对象调用,同样,在block方法体内,可以通过this来调用到receiver。with返回的类型为R,和block的返回类型相同

fun getPersonFromDeveloper(developer: Developer): Person {
    return with(developer) {
        Person(developerName, developerAge)
    }
}

参考:
learn kotlin apply vs with
what is a receiver in kotlin
What is a purpose of Lambda's with Receiver?

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

推荐阅读更多精彩内容