Answers:
fold
接受一个初始值,传递给它的lambda的第一次调用将接收该初始值和该集合的第一个元素作为参数。
例如,采用以下代码来计算整数列表的总和:
listOf(1, 2, 3).fold(0) { sum, element -> sum + element }
对lambda的第一个调用将带有参数0
和1
。
如果必须为操作提供某种默认值或参数,则可以传递初始值很有用。例如,如果您正在列表中寻找最大值,但由于某种原因想要返回至少10,则可以执行以下操作:
listOf(1, 6, 4).fold(10) { max, element ->
if (element > max) element else max
}
reduce
不需要初始值,而是从集合的第一个元素开始作为累加器(sum
在以下示例中调用)。
例如,让我们再次对整数求和:
listOf(1, 2, 3).reduce { sum, element -> sum + element }
这里对lambda的第一个调用将带有参数1
和2
。
reduce
当您的操作除了要应用到的集合中的值不依赖于任何其他值时,可以使用。
emptyList<Int>().reduce { acc, s -> acc + s }
将产生异常,但是emptyList<Int>().fold(0) { acc, s -> acc + s }
可以。
listOf<Int>(1, 2).reduce { acc: Number, i: Int -> acc.toLong() + i }
(列表类型为Int,而累加器类型声明为Number且实际上为Long)
我要指出的主要功能差异(在其他答案的注释中提到,但可能很难理解)是,如果对空集合执行此操作reduce
将引发异常。
listOf<Int>().reduce { x, y -> x + y }
// java.lang.UnsupportedOperationException: Empty collection can't be reduced.
这是因为.reduce
不知道在“无数据”的情况下要返回什么值。
将此与进行对比.fold
,这需要您提供一个“起始值”,如果出现空集合,它将是默认值:
val result = listOf<Int>().fold(0) { x, y -> x + y }
assertEquals(0, result)
因此,即使您不想将集合汇总为其他类型(不相关)的单个元素(只会.fold
让您这样做),如果起始集合可能为空,那么您也必须检查集合先设置大小,然后.reduce
使用.fold
val collection: List<Int> = // collection of unknown size
val result1 = if (collection.isEmpty()) 0
else collection.reduce { x, y -> x + y }
val result2 = collection.fold(0) { x, y -> x + y }
assertEquals(result1, result2)