kotlin - 如何使用 Kotlin 的谓词将列表拆分为子列表?

标签 kotlin functional-programming

我正在尝试一种惯用的、理想的功能性方法来将 Kotlin 中的列表拆分为子列表。

假设输入是["aaa", "bbb", "", "ccc", "", "ddd", "eee", "fff"],我想返回[["aaa", "bbb"], ["ccc"], ["ddd", "eee", "fff"]] 给定谓词 string.isEmpty()

使用 for 循环和累加器非常简单;但我还没有找到一种我认为足够可读的功能性编写方法。

到目前为止,我最好的结果是:

lines.foldIndexed(Pair(listOf<List<String>>(), listOf<String>()), { idx, acc, line ->
    when {
    idx + 1 == lines.size -> {
        Pair(acc.first + listOf(acc.second + line), listOf())
    }
    line.isEmpty() -> {
        Pair(acc.first + listOf(acc.second), listOf())
    }
    else -> {
        Pair(acc.first, acc.second + line)
        }
    }
}).first

本质上,我使用的是带有双累加器的 fold,它跟踪当前列表并在找到谓词时重置。该列表会在此时输入完整的结果。我正在使用 foldIndexed为了得到我的最后一个列表。

大家有什么更好的办法吗?

作为引用,循环版本可以是

val data = mutableListOf<String>()
var currentData = ""
for(line in lines){
    if(line.isEmpty()) {
        data.add(currentData)
        currentData = ""
    }
    else{
        currentData = "$currentData $line"
    }
}
data.add(currentData)

谢谢!

最佳答案

我建议先找到 split 点(手动添加边缘索引),然后再做切片:

val lines = listOf("aaa", "bbb", "", "ccc", "", "ddd", "eee", "fff")
val result = lines
    .flatMapIndexed { index, x ->
        when {
            index == 0 || index == lines.lastIndex -> listOf(index)
            x.isEmpty() -> listOf(index - 1, index + 1)
            else -> emptyList()
        }
    }
    .windowed(size = 2, step = 2) { (from, to) -> lines.slice(from..to) }
println(result) //[[aaa, bbb], [ccc], [ddd, eee, fff]]

关于kotlin - 如何使用 Kotlin 的谓词将列表拆分为子列表?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/65248942/

相关文章:

parsing - 以功能纯方式生成不可变的具体语法树的适当数据结构或算法是什么?

typescript - OO 到函数式——从日常问题中学习

kotlin - 当 "when"语句未涵盖所有实现时,强制编译器发出错误

kotlin - 不清楚sumBy block 中的参数如何工作(Kotlin)

android - 保存后个人资料图片不显示(它只显示空白屏幕)

generics - 为什么我收到此代码的 'receiver type mismatch' 错误

javascript - 使用 Gradle 在 IntelliJ 中使用多个源集的 2 个项目 Unresolved 依赖关系

javascript - 参数从哪里来?

scala - 在 scalaz 中组合状态的标准方法

functional-programming - "Stacking"函数式语言中的有效函数 - DrRacket