繁体   English   中英

如何在kotlin中组合两个不同长度的列表?

[英]How to combine two different length lists in kotlin?

我想合并两个不同长度的列表。 例如;

val list1 = listOf(1,2,3,4,5)
val list2 = listOf("a","b","c")

我想要这样的结果

(1,"a",2,"b",3,"c",4,5)

有什么建议吗?

您可以为此使用.zip函数

list1.zip(list2){ a,b -> listOf(a,b)}.flatten()

唯一的问题是它只会处理两个集合的元素,所以如果(就像在例子中一样)让我们有不同的大小 - 它不会工作

另一种方法是添加特定的标记并过滤它们,或者只使用迭代器。 我找到了一个带有sequence{..}函数的优雅解决方案

 val result = sequence {
    val first = list1.iterator()
    val second = list2.iterator()
    while (first.hasNext() && second.hasNext()) {
      yield(first.next())
      yield(second.next())
    }

    yieldAll(first)
    yieldAll(second)
  }.toList()
  1. 如果源列表中的元素可以在结果列表中以任何顺序出现,则
>>> list1 + list2
res12: kotlin.collections.List<kotlin.Any> = [1, 2, 3, 4, 5, a, b, c]
  1. 如果源列表中的元素在结果列表中交替出现并且 list1 比 list2 长,则
>>> list1.zip(list2).flatMap { listOf(it.first, it.second) } + list1.drop(list2.size)
res16: kotlin.collections.List<kotlin.Any> = [1, a, 2, b, 3, c, 4, 5]

你可以这样做:

val mergedList = with(setOf(list1, list2).sortedByDescending { it.count() }) {
    first().mapIndexed { index, e ->
        listOfNotNull(e, last().getOrNull(index))
    }
}.flatten()

首先,您将两个列表放在一个Set ,然后按元素数量对其进行排序(降序),生成列表列表。

第一个列表具有最多的元素将用于迭代。

使用mapIndexed您可以使用index访问第二个列表中的相应元素。 如果没有,则返回null并被listOfNotNull过滤掉。 最后,您将生成的列表列表展平,并获得所需的结果:

[1, a, 2, b, 3, c, 4, 5]

我认为 Eugenes 的答案已经包含了组合两个列表(无论是zip还是组合所有元素)所需的所有知识。

如果您想组合任意数量的列表,每个交替列表一个项目,您可能还对以下方法感兴趣:

fun combine(vararg lists: List<*>) : List<Any> = mutableListOf<Any>().also {
  combine(it, lists.map(List<*>::iterator))
}

private tailrec fun combine(targetList: MutableList<Any>, iterators: List<Iterator<*>>) {
  iterators.asSequence()
          .filter(Iterator<*>::hasNext)
          .mapNotNull(Iterator<*>::next)
          .forEach { targetList += it }
  if (iterators.asSequence().any(Iterator<*>::hasNext))
    combine(targetList, iterators)
}

调用它然后看起来如下并导致在评论中看到的值:

combine(list1, list2) // List containing: 1, "a", 2, "b", 3, "c", 4, 5
combine(list1, list2, listOf("hello", "world")) // 1, "a", "hello", 2, "b", "world", 3, "c", 4, 5

可以使用以下代码实现 Eugenes 答案第二部分的简化方法; 当然,当你得到一个列表时,它不再懒惰了;-)(但也许你甚至直接将它翻译成一个列表,所以你也可以使用这种方法):

fun List<Any>.combine(other: List<Any>) : List<Any> = mutableListOf<Any>().also {
  val first = iterator()
  val second = other.iterator()
  while (first.hasNext() || second.hasNext()) {
    if (first.hasNext()) it.add(first.next())
    if (second.hasNext()) it.add(second.next())
  }
}

调用它的工作方式如下:

list1.combine(list2)

这是我的看法:

fun <T> zip(vararg iterables: Iterable<T>): List<T> = iterables
.map { it.iterator() }
.toList()
.let { iterators ->
    mutableListOf<T>()
        .also { list ->
            while (
              iterators.any {if (it.hasNext()) list.add(it.next()) else false }
            ) { }
        }
}

您的列表是不可转换的类型(整数和字符串),因此您必须拥有MutableList<Any>以便您可以添加这两种类型:

val allItems = mutableListOf<Any>(1,2,3,4,5)
val list2 = listOf("a","b","c")
allItems.addAll(list2)

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM