Kotlin中有多个variables
有没有办法链接多个让kotlin中的多个可为空的variables?
fun example(first: String?, second: String?) { first?.let { second?.let { // Do something just if both are != null } } }
我的意思是,这样的事情:
fun example(first: String?, second: String?) { first?.let && second?.let { // Do something just if both are != null } }
这里有一些变化,取决于你想要使用什么风格,如果你有一切相同或不同的types,如果名单未知数量的项目…
混合types,不能为null来计算新的值
对于混合types,您可以为每个参数计数建立一系列函数,这些函数可能看起来很愚蠢,但对于混合types很适用:
fun safeLet(p1: T1?, p2: T2?, block: (T1, T2)->R?): R? { return if (p1 != null && p2 != null) block(p1, p2) else null } fun safeLet(p1: T1?, p2: T2?, p3: T3?, block: (T1, T2, T3)->R?): R? { return if (p1 != null && p2 != null && p3 != null) block(p1, p2, p3) else null } fun safeLet(p1: T1?, p2: T2?, p3: T3?, p4: T4?, block: (T1, T2, T3, T4)->R?): R? { return if (p1 != null && p2 != null && p3 != null && p4 != null) block(p1, p2, p3, p4) else null } fun safeLet(p1: T1?, p2: T2?, p3: T3?, p4: T4?, p5: T5?, block: (T1, T2, T3, T4, T5)->R?): R? { return if (p1 != null && p2 != null && p3 != null && p4 != null && p5 != null) block(p1, p2, p3, p4, p5) else null } // ...keep going up to the parameter count you care about
用法示例:
val risk = safeLet(person.name, person.age) { name, age -> // do something }
当列表中没有空项目时,执行代码块
这里有两种风格,第一种是在列表中包含所有非空项目时执行代码块,第二种在列表中至少有一个非空项目时执行相同操作。 两种情况都将非空项目列表传递给代码块:
function:
fun Collection.whenAllNotNull(block: (List)->R) { if (this.all { it != null }) { block(this.filterNotNull()) } } fun Collection.whenAnyNotNull(block: (List)->R) { if (this.any { it != null }) { block(this.filterNotNull()) } }
用法示例:
listOf("something", "else", "matters").whenAllNotNull { println(it.joinToString(" ")) } // output "something else matters" listOf("something", null, "matters").whenAllNotNull { println(it.joinToString(" ")) } // no output listOf("something", null, "matters").whenAnyNotNull { println(it.joinToString(" ")) } // output "something matters"
让函数获得项目列表并做相同的操作稍作改动:
fun whenAllNotNull(vararg options: T?, block: (List)->R) { if (options.all { it != null }) { block(options.filterNotNull()) } } fun whenAnyNotNull(vararg options: T?, block: (List)->R) { if (options.any { it != null }) { block(options.filterNotNull()) } }
用法示例:
whenAllNotNull("something", "else", "matters") { println(it.joinToString(" ")) } // output "something else matters"
这些变化可以被改变成像let()
那样的返回值。
使用第一个非空项目(合并)
与SQL合并函数类似,返回第一个非空项目。 两种风格的function:
fun coalesce(vararg options: T?): T? = options.firstOrNull { it != null } fun Collection.coalesce(): T? = this.firstOrNull { it != null }
用法示例:
coalesce(null, "something", null, "matters")?.let { it.length } // result is 9, length of "something" listOf(null, "something", null, "matters").coalesce()?.let { it.length } // result is 9, length of "something"
其他变化
…还有其他的变化,但更多的规格可以缩小。
你可以写下你自己的function:
fun Pair.biLet(body: (T, U) -> R): R? { if (first != null && second != null) { return body(first, second) } return null } (first to second).biLet { first, second -> // body }
根据Kotlin的各种*NotNull
和*OrNull
函数的精神,您可以创建一个arrayOfNotNullOrNull
函数,以允许您在所有variables不为null
创建一个“多个可为空的variables”的数组,否则为null
:
fun arrayOfNotNullOrNull(vararg elements: T?): Array ? { for (element in elements) { if (element == null) { return null } } return elements as Array }
然后,您可以使用let
来获取可变数目的值:
fun example(first: String?, second: String?) { arrayOfNotNullOrNull(first, second)?.let { // Do something just if both are != null // eg val (notNullFirst, notNullSecond) = it ... } }
如果你已经有一个集合的空值,那么你可以创建一个类似于kotlin.collections.requireNoNulls
的noNullsOrNull
扩展函数,但是返回null
而不是抛出一个exception。
其实,你可以简单地做到这一点,你知道吗? ;)
if (first != null && second != null) { // your logic here... }
在Kotlin中使用普通的空检查没有任何问题。
对于那些会查看你的代码的人来说,它更具可读性。
对于只检查两个值的情况,也不需要使用列表:
fun ifNotNull(value1: T1?, value2: T2?, bothNotNull: (T1, T2) -> (Unit)) { if (value1 != null && value2 != null) { bothNotNull(value1, value2) } }
用法示例:
var firstString: String? var secondString: String? ifNotNull(firstString, secondString) { first, second -> Log.d(TAG, "$first, $second") }