gpt4 book ai didi

kotlin - 在 Kotlin 中处理可为空或空列表的惯用方式

转载 作者:IT老高 更新时间:2023-10-28 13:28:27 28 4
gpt4 key购买 nike

假设我有一个变量 activities类型 List<Any>? .如果列表不为空且不为空,我想做点什么,否则我想做点别的。我想出了以下解决方案:

when {
activities != null && !activities.empty -> doSomething
else -> doSomethingElse
}

在 Kotlin 中有没有更惯用的方法来做到这一点?

最佳答案

对于一些简单的操作,您可以使用安全调用运算符,假设该操作还尊重不在空列表上的操作(处理您的 both null 和 empty 的情况:

myList?.forEach { ...only iterates if not null and not empty }

用于其他操作。你可以编写一个扩展函数——两种变体取决于你是想将列表作为 this 还是作为参数接收:

inline fun <E: Any, T: Collection<E>> T?.withNotNullNorEmpty(func: T.() -> Unit): Unit {
if (this != null && this.isNotEmpty()) {
with (this) { func() }
}
}

inline fun <E: Any, T: Collection<E>> T?.whenNotNullNorEmpty(func: (T) -> Unit): Unit {
if (this != null && this.isNotEmpty()) {
func(this)
}
}

你可以用作:

fun foo() {  
val something: List<String>? = makeListOrNot()
something.withNotNullNorEmpty {
// do anything I want, list is `this`
}

something.whenNotNullNorEmpty { myList ->
// do anything I want, list is `myList`
}
}

你也可以做反函数:

inline fun <E: Any, T: Collection<E>> T?.withNullOrEmpty(func: () -> Unit): Unit {
if (this == null || this.isEmpty()) {
func()
}
}

我会避免将这些链接起来,因为那样您将用更冗长的内容替换 ifwhen 语句。而且您正在更多地进入我在下面提到的替代方案提供的领域,这是成功/失败情况的完整分支。

注意:这些扩展被推广到所有持有非空值的 Collections 的后代。并且不仅仅为列表工作。

替代方案:

Result Kotlin 库提供了一种很好的方法来处理基于响应值的“做这个或那个”的情况。对于 Promises,您可以在 Kovenant 中找到相同的内容。图书馆。

这两个库都为您提供了从单个函数返回替代结果的方式,也为您提供了基于结果分支代码的方式。 他们确实要求您控制所采取行动的“答案”的提供者。

这些是 OptionalMaybe 的良好 Kotlin 替代品。

进一步探索扩展功能(也许太多了)

本节只是为了表明,当您遇到类似于此处提出的问题的问题时,您可以在 Kotlin 中轻松找到许多答案,从而按照您想要的方式进行编码。如果这个世界不讨人喜欢,那就改变这个世界。它不是一个好或坏的答案,而是额外的信息。

如果您喜欢扩展函数并想考虑将它们链接到表达式中,我可能会将它们更改如下...

返回 thiswithXyz 风格和 whenXyz 应该返回一种新类型,允许整个集合变成新的类型(甚至与原著无关)。产生如下代码:

val BAD_PREFIX = "abc"
fun example(someList: List<String>?) {
someList?.filterNot { it.startsWith(BAD_PREFIX) }
?.sorted()
.withNotNullNorEmpty {
// do something with `this` list and return itself automatically
}
.whenNotNullNorEmpty { list ->
// do something to replace `list` with something new
listOf("x","y","z")
}
.whenNullOrEmpty {
// other code returning something new to replace the null or empty list
setOf("was","null","but","not","now")
}
}

注意:此版本的完整代码在文章末尾(1)

但您也可以使用自定义的“否则那样”机制走向全新的方向:

fun foo(someList: List<String>?) {
someList.whenNullOrEmpty {
// other code
}
.otherwise { list ->
// do something with `list`
}
}

没有限制,要有创意,学习扩展的力量,尝试新的想法,正如您所见,人们对这类情况的编码方式有很多变化。标准库不能不混淆地支持这些类型的方法的 8 种变体。但是每个开发组都可以拥有与其编码风格相匹配的扩展。

注意:此版本的完整代码在文章末尾(2)

示例代码 1: 这是“链式”版本的完整代码:

inline fun <E: Any, T: Collection<E>> T?.withNotNullNorEmpty(func: T.() -> Unit): T? {
if (this != null && this.isNotEmpty()) {
with (this) { func() }
}
return this
}

inline fun <E: Any, T: Collection<E>, R: Any> T?.whenNotNullNorEmpty(func: (T) -> R?): R? {
if (this != null && this.isNotEmpty()) {
return func(this)
}
return null
}

inline fun <E: Any, T: Collection<E>> T?.withNullOrEmpty(func: () -> Unit): T? {
if (this == null || this.isEmpty()) {
func()
}
return this
}

inline fun <E: Any, T: Collection<E>, R: Any> T?.whenNullOrEmpty(func: () -> R?): R? {
if (this == null || this.isEmpty()) {
return func()
}
return null
}

示例代码 2: 这是“this else that”库的完整代码(带有单元测试):

inline fun <E : Any, T : Collection<E>> T?.withNotNullNorEmpty(func: T.() -> Unit): Otherwise {
return if (this != null && this.isNotEmpty()) {
with (this) { func() }
OtherwiseIgnore
} else {
OtherwiseInvoke
}
}

inline fun <E : Any, T : Collection<E>> T?.whenNotNullNorEmpty(func: (T) -> Unit): Otherwise {
return if (this != null && this.isNotEmpty()) {
func(this)
OtherwiseIgnore
} else {
OtherwiseInvoke
}
}

inline fun <E : Any, T : Collection<E>> T?.withNullOrEmpty(func: () -> Unit): OtherwiseWithValue<T> {
return if (this == null || this.isEmpty()) {
func()
OtherwiseWithValueIgnore<T>()
} else {
OtherwiseWithValueInvoke(this)
}
}

inline fun <E : Any, T : Collection<E>> T?.whenNullOrEmpty(func: () -> Unit): OtherwiseWhenValue<T> {
return if (this == null || this.isEmpty()) {
func()
OtherwiseWhenValueIgnore<T>()
} else {
OtherwiseWhenValueInvoke(this)
}
}

interface Otherwise {
fun otherwise(func: () -> Unit): Unit
}

object OtherwiseInvoke : Otherwise {
override fun otherwise(func: () -> Unit): Unit {
func()
}
}

object OtherwiseIgnore : Otherwise {
override fun otherwise(func: () -> Unit): Unit {
}
}

interface OtherwiseWithValue<T> {
fun otherwise(func: T.() -> Unit): Unit
}

class OtherwiseWithValueInvoke<T>(val value: T) : OtherwiseWithValue<T> {
override fun otherwise(func: T.() -> Unit): Unit {
with (value) { func() }
}
}

class OtherwiseWithValueIgnore<T> : OtherwiseWithValue<T> {
override fun otherwise(func: T.() -> Unit): Unit {
}
}

interface OtherwiseWhenValue<T> {
fun otherwise(func: (T) -> Unit): Unit
}

class OtherwiseWhenValueInvoke<T>(val value: T) : OtherwiseWhenValue<T> {
override fun otherwise(func: (T) -> Unit): Unit {
func(value)
}
}

class OtherwiseWhenValueIgnore<T> : OtherwiseWhenValue<T> {
override fun otherwise(func: (T) -> Unit): Unit {
}
}


class TestBrancher {
@Test fun testOne() {
// when NOT null or empty

emptyList<String>().whenNotNullNorEmpty { list ->
fail("should not branch here")
}.otherwise {
// sucess
}

nullList<String>().whenNotNullNorEmpty { list ->
fail("should not branch here")
}.otherwise {
// sucess
}

listOf("a", "b").whenNotNullNorEmpty { list ->
assertEquals(listOf("a", "b"), list)
}.otherwise {
fail("should not branch here")
}

// when YES null or empty

emptyList<String>().whenNullOrEmpty {
// sucess
}.otherwise { list ->
fail("should not branch here")
}

nullList<String>().whenNullOrEmpty {
// success
}.otherwise {
fail("should not branch here")
}

listOf("a", "b").whenNullOrEmpty {
fail("should not branch here")
}.otherwise { list ->
assertEquals(listOf("a", "b"), list)
}

// with NOT null or empty

emptyList<String>().withNotNullNorEmpty {
fail("should not branch here")
}.otherwise {
// sucess
}

nullList<String>().withNotNullNorEmpty {
fail("should not branch here")
}.otherwise {
// sucess
}

listOf("a", "b").withNotNullNorEmpty {
assertEquals(listOf("a", "b"), this)
}.otherwise {
fail("should not branch here")
}

// with YES null or empty

emptyList<String>().withNullOrEmpty {
// sucess
}.otherwise {
fail("should not branch here")
}

nullList<String>().withNullOrEmpty {
// success
}.otherwise {
fail("should not branch here")
}

listOf("a", "b").withNullOrEmpty {
fail("should not branch here")
}.otherwise {
assertEquals(listOf("a", "b"), this)
}


}

fun <T : Any> nullList(): List<T>? = null
}

关于kotlin - 在 Kotlin 中处理可为空或空列表的惯用方式,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26341225/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com