Kotlin高阶函数使用

Lambda表达式

说到高阶函数,不得不先接受下Lambda表达式,一个完整的lambda表达式如下:

fun main(){
    //lambda表达式
    var method03 : (Int,Int) -> Int = {
        i:Int,j:Int -> i+j
    }
    println("method03 return: ${method03(1,2)}")
}

声明method03是一个函数类型,而该函数类型的实例就是=号后面的部分,花括号包含的部分就是lambda表达式,用lambda表达式来表示一个函数类型实例

看下以下几种写法:

fun main(){
    //可以仅声明一个函数类型而没有赋值,但无法调用
    var method01 : () -> Unit
    var method02 : (Int,Int) -> Int

    //完整函数声明
    var method03 : (Int,Int) -> Int = {
        i:Int,j:Int -> i+j
    }
    println("method03 return: ${method03(1,2)}")
    
    //省略函数类型声明,在lambda表达式中可确定,返回类型根据num1 + num2推断
    var method04 = {num1 : Int, num2 : Int -> num1 + num2}
    println("method04 return: ${method04(4,2)}")
    
    //无入参和返回值
    var method05 = { println("method05:无入参和返回值")
    }
    method05()
    
    //有入参,无返回值,省略函数类型声明
    var method06 = { num1 : Int -> println("method06:$num1")}
    method06(1)
    
    //无入参,有返回值,省略函数类型声明
    var method07 = { 1 }
    println("method7 return: ${method07()}")
    
    //有入参,由于有函数类型声明,lambda内部入参可以省略类型
    var method08 :(Int) -> Unit = {
        x ->
        when (x) {
            1 -> println("这个数是1")
        }
    }
    method08(1)
    //当只有一个参数时可省略,即是lambda表达式中的it
    var method09 :(Int) -> Unit = {
        when (it) {
            2 -> println("method09:这个数是2")
        }
    }
    method09(2)
}

运行结果如下:
image.png

高阶函数

高阶函数其实就是将函数用作参数或返回值的函数,接受了前面的lambda表达式,就可以知道一个函数其实也是一个对象,函数也可以作为一个参数

先看个简单代码:

fun main(){
//    method(true,{
//        println("调用此函数")
//    })
    method(true){
        println("调用此函数")
    }
}
fun method(b: Boolean, m: () -> Unit){
    if (b) {
        m()
    }
}

注:最后一个参数是 lambda 表达式时,那么它既可作为参数在括号内传入,也可以在括号外传入
method方法接收两个参数,一个布尔类型,一个是函数类型的参数,此method即为高阶函数,代码可能很简单,并没有什么实际意义,只是为了展示高阶函数表达方式

有了上面的基础,下面看一个有意义的例子:
lateinit var adapter : Adapter
fun main(){
    val context: Context? = null
//    val adapter: Adapter = Adapter(context!!)
    val adapter: Adapter? = context?.myWith { Adapter(it) }
}
fun <T,R> T.myWith(createAdapter : (T) -> R) : R{
    return createAdapter(this)
}

上面创建一个适配器传入context,是我们Android中常用的代码,但是adapter是我们Android的,所以传入context时,编译器要求这样val adapter: Adapter = Adapter(context!!),但这样写不太优雅,而又不想去if判空,就可以增加一个如myWith的扩展函数:

  • <T,R>T,R泛型分别代表入参类型和返回值类型;
  • T.myWith是为T类型增加一个扩展函数myWith
  • 入参是一个lambda表达式,也就是函数类型createAdapter : (T) -> R,这个函数的入参是T类型,返回值是R类型
  • : R表示myWith方法的返回值也是R类型
  • 因为myWith是T的扩展函数,调用createAdapter传入的this即是T类型,故context?.myWith { Adapter(it) }中的it即是T类型,所以createAdapter(this)返回的就是Adapter(context),myWith返回的就是createAdapter(this)返回的结果也是Adapter(context)
最后再看一个例子:
var name: String = "xiaoming"
fun main(){
    name.let1 {
        println(it)
    }
    name.let2{
        println(this)
    }
}
fun <T,R> T.let1(m : (T) -> R) : R{
    return m(this)
}
fun <T,R> T.let2(m : T.() -> R) : R{
    return m()
}

运行结果:
image.png

为所有类型增加let1和let2扩展函数:let1扩展函数其实已经说过了,主要看下let2,
形参m的函数类型不一样:T.() -> RT.()是为T类增加了一个匿名函数,而这个匿名函数,它的函数实例是name.let2{ println(this) }中传入的{ println(this) }参数,所以这个lambda表达式函数可以直接访问到T,无需通过传参。

注:上面的例子其实都是模仿kotlin源码中的扩展函数如let,with,还有很多apply,run,repeat等扩展函数,感兴趣的可以自行翻看源码

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