Lambda就是一小段可以作为参数传递的代码,在kotlin中的标准形式为:
{参数名1: 参数类型, 参数名2: 参数类型 -> 函数体}
标准形式使用如下:
val list = listOf("Apple", "Banana", "Orange", "Pear", "Grape", "Watermelon")
val lambda = { fruit: String -> fruit.length }
val maxLengthFruit = list.maxBy(lambda)
上面的例子中我们使用了一个变量lambda来记录Lambda表达式,实际上我们可以省略这个变量,直接将表达式传入函数中
val maxLengthFruit = list.maxBy({ fruit: String -> fruit.length })
当Lambda参数是函数的最后一个参数时,可以将Lambda表达式移到函数括号的外面
val maxLengthFruit = list.maxBy() { fruit: String -> fruit.length }
如果Lambda参数是函数的唯一一个参数的话,还可以将函数的括号省略
val maxLengthFruit = list.maxBy { fruit: String -> fruit.length }
由于Kotlin拥有出色的类型推导机制,Lambda表达式中的参数列表其实在大多数情况下不必声明参数类型
val maxLengthFruit = list.maxBy { fruit -> fruit.length }
当Lambda表达式的参数列表中只有一个参数时,也不必声明参数名,而是可以使用it关键字来代替
val maxLengthFruit = list.maxBy { it.length }
在实际的使用中,可以根据具体的情况采用不同的表达形式
Kotlin在编译时进行判空检查,默认所有的参数和变量都不为空, 如果需要使用null作为参数,那么需要采用另外一套可为空的参数系统.
如:Int? String? Double?
当对象不为空时正常调用相应的方法,当对象为空时则什么都不做,例如:
if (a != null) {
a.doSomething()
}
可以简化为:
a?.doSomething()
这个操作符的左右两边都接收一个表达式,如果左边表达式的结果不为空就返回左边表达式的结果,否则就返回右边表达式的结果.
val c = if (a ! = null) {
a
} else {
b
}
这段代码的逻辑使用?:操作符就可以简化成:
val c = a ?: b
fun getTextLength(text: String?): Int {
if (text != null) {
return text.length
}
return 0
}
可以简化为:
fun getTextLength(text: String?) = text?.length ?: 0
首先由于text是可能为空的,因此我们在调用它的length字段时需要使用?.操作符,而当text为空时,text?.length会返回一个null值,这个时候我们再借助?:操作符让它返回0.
在对象的后面加上!!,表示不需要kotlin在此进行非空检查,慎用!!
这个函数提供了函数式API的编程接口,并将原始调用对象作为参数传递到Lambda表达式中。示例代码如下:
obj.let { obj2 ->
// 编写具体的业务逻辑
}
这里调用了obj对象的let函数,然后Lambda表达式中的代码就会立即执行,并且这个obj对象本身还会作为参数传递到Lambda表达式中。不过,为了防止变量重名,这里将参数名改成了obj2,但实际上它们是同一个对象,这就是let函数的作用.
let函数的特性配合?.操作符可以在空指针检查的时候起到很大的作用
fun doStudy(study: Study?) {
study?.let { stu ->
stu.readBooks()
stu.doHomework()
}
}
?.操作符表示对象为空时什么都不做,对象不为空时就调用let函数,而let函数会将study对象本身作为参数传递到Lambda表达式中,此时的study对象肯定不为空了,我们就能放心地调用它的任意方法了.
当Lambda表达式的参数列表中只有一个参数时,可以不用声明参数名,直接使用it关键字来代替即可,那么代码就可以进一步简化成:
fun doStudy(study: Study?) {
study?.let {
it.readBooks()
it.doHomework()
}
}
let函数是可以处理全局变量的判空问题的,而if判断语句则无法做到这一点。因为全局变量的值随时都有可能被其他线程所修改,即使做了判空处理,仍然无法保证if语句中的study变量没有空指针风险。从这一点上也能体现出let函数的优势。
kotlin在定义函数的时候给任意参数设定一个默认值,这样当调用此函数时就不会强制要求调用方为此参数传值,在没有传值的情况下会自动使用参数的默认值.
fun printParams(num: Int, str: String = "hello") {
println("num is $num , str is $str")
}
当调用printParams()函数时,可以选择给第二个参数传值,也可以选择不传,在不传的情况下就会自动使用默认值.
如果是第一个参数设定了默认值呢,那我们不想传第一个参数时怎么办呢,如果直接传递第二个参数将会报类型不匹配的错误,这个时候需要使用键值对的方式来进行传参.比如调用printParams()函数,我们还可以这样写:
printParams(str = "world", num = 123)
此时哪个参数在前哪个参数在后都无所谓,Kotlin可以准确地将参数匹配上.