Kotlin Lambda

目录

引言

定义

语法结构

拓展

Java函数式API

举例

拓展


引言

先看一段代码,其通过listOf快速创建一个字符串(液体)集合

val list = listOf("water", "milk", "juice", "soup", "gasoline")

如果想要获取这个集合中的字符长度最长的一项,应该怎么实现呢?

常规写法:

var maxWord = "";
for (word in list) {
    if (word.length > maxWord.length) {
        maxWord = word
    }
}

而通过集合的函数式API可以更轻松的实现,只用了一行代码

val maxWord = list.maxBy { it.length }

分析:maxBy就是一个普通的函数而已,它接收一个Lambda类型参数,在遍历集合时将每次遍历的值作为参数传递给Lambda表达式。maxBy函数的工作原理是根据传入的条件来遍历集合,从而找到该条件下的最大值,比如说想要找到单词长度最长的液体,条件自然就应该是单词的长度了。那么什么是Lambda呢?

定义

        用最直白的语言来阐述的话,Lambda就是一小段可以作为参数传递的代码。通常不建议在Lambda表达式中编写太长的代码,否则可能会影响代码的可读性。

语法结构

{参数名1: 参数类型, 参数名2: 参数类型,…… -> 函数体}

       首先最外层是一对大括号,如果有参数传入到 Lambda表达式中的话,我们还需要声明参数列表,参数列表的结尾使用一个->符号,表示参数列表的结束以及函数体的开始,函数体中可以编写任意行代码(虽然不建议编写太长的代码),并且最后一行代码会自动作为Lambda表达式的返回值。

        那么为什么引言中maxBy函数中传入的Lambda表达式似乎并不是这个格式呢?

        因为那是Lambda的简化写法。虽然很多情况下,开发时并不需要使用Lambda表达式完整的语法结构,但是对于初学者来说,简化版写法不易理解,下面进行推导演化看看简化版写法从何而来。

val lambda = { word: String -> word.length }
val str = list.maxBy(lambda)
        /*
         |
         |    Lambda表达式不用专门定义一个变量
         |
         v
         */
val str2 = list.maxBy({ word: String -> word.length })
        /*
         |
         |    Kotlin规定,当Lambda参数是函数的最后一个参数时,
         |    可以将Lambda表达式移到函数括号的外面
         |
         v
         */
val str3 = list.maxBy() { word: String -> word.length }
        /*
         |
         |    如果Lambda参数是函数的唯一一个参数的话,还可以将函数的括号省略
         |
         v
         */
val str4 = list.maxBy { word: String -> word.length }
        /*
         |
         |    由于Kotlin拥有出色的类型推导机制,
         |    Lambda表达式中的参数列表其实在大多数情况下不必声明参数类型    
         |
         v
        */
val str5 = list.maxBy { word -> word.length }
        /*
         |
         |    当Lambda表达式的参数列表中只有一个参数时,
         |    不必声明参数名,而是可以使用it关键字来代替
         |
         v
        */
val str6 = list.maxBy { it.length }

拓展

        Kotlin中除了maxBy,还有一些常见的函数式API,如:map,filter,any,all……

通过组合mapfilter实现将前文例子中的液体集合

  • 过滤小于5个字母的单词
  • 单词转换为大写
val list = listOf("water", "milk", "juice", "soup", "gasoline")
val filterList = list.filter { it.length >= 5 }.map { it.uppercase() }

        先调用filter会比先调用map更加高效。通过调用filter后集合的长度变短了,后续map需要遍历的次数随之减少。


        any函数用于判断集合中是否至少存在一个元素满足指定条件,all函数用于判断集合中是否所有元素都满足指定条件。

var anyResult = list.any { it.length > 5 }  //Boolean
var allResult = list.all { it.length < 4 }  //Boolean

Java函数式API

        上文介绍了Kotlin中函数式API的用法,但实际上在Kotlin中调用Java方法时也可以使用函数式API,只不过这是有一定条件限制的。

        具体来讲,如果我们在Kotlin代码中调用了一个Java方法,并且该方法接收一个Java单抽象方法接口参数,就可以使用函数式API。Java单抽象方法接口指的是接口中只有一个待实现方法,如果接口中有多个待实现方法,则无法使用函数式API。

        因为Android SDK还是使用Java语言编写的,当我们在Kotlin中调用这些SDK接口时,就很可能会用到这种Java函数式API的写法。

举例

public interface Runnable {
    void run();
}

/*
 Kotlin中创建线程并传入Runnable参数,
 Kotlin中完全舍弃了new关键字,因此创建匿名内部类时使用object关键字
 和Java写法似乎没有什么简化之处
*/
Thread(object : Runnable {
    override fun run() {
    ...
    }
}).start()

|
|
v

/*
 因为Runnable类中只有一个待实现方法,即使这里没有显式地重写run()方法,
 Kotlin也能自动明白Runnable后面的Lambda表达式就是要在run()方法中实现的内容
*/
Thread(Runnable {
    ...     
}).start()

|
|
v

/*
 如果一个Java方法的参数列表中有且仅有一个Java单抽象方法接口参数,
 我们还可以将接口名进行省略
*/
Thread({
    ...      
}).start()

|
|
v

/*
 当Lambda表达式是方法的最后一个参数时,可以将Lambda表达式移到方法括号的外面。
 同时,如果Lambda表达式还是方法的唯一一个参数,还可以将方法的括号省略
*/
Thread {...}.start()

拓展

按钮注册点击事件

findViewById<Button>(R.id.btn).setOnClickListener({v: View -> 
    Toast.makeText(this, "点击", Toast.LENGTH_SHORT).show()
})

|
|
v

btn.setOnClickListener({
    Toast.makeText(this, "点击", Toast.LENGTH_SHORT).show()
})

|
|
v

btn.setOnClickListener(){
    Toast.makeText(this, "点击", Toast.LENGTH_SHORT).show()
}

|
|
v

btn.setOnClickListener{
    Toast.makeText(this, "点击", Toast.LENGTH_SHORT).show()
}

  • 20
    点赞
  • 18
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值