目录
引言
先看一段代码,其通过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……
通过组合map,filter实现将前文例子中的液体集合
- 过滤小于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()
}