我有以下代码,该代码将满足谓词p
的任何元素之前的数组拆分为数组。它输入检查:
def splitBefore[T](a: Array[T], p: (T) => Boolean)
(implicit tct: ClassTag[T]): Array[Array[T]] =
a.foldLeft(Array[Array[T]](Array.empty[T])) {
(acc: Array[Array[T]], s: T) => if (p(s))
acc :+ Array(s)
else
acc.init :+ (acc.last :+ s)
}
当我使用非空的
a
调用它时,它可以正常工作:scala> splitBefore(Array("a", "BC", "d"), (s: String) => s.size > 1)
res1: Array[Array[String]] = Array(Array(a), Array(BC, d))
但是当我用一个空数组调用它时,我得到了
ClassCastException
:scala> splitBefore(Array.empty[String], (s: String) => s.size > 1)
java.lang.ClassCastException: [Ljava.lang.Object; cannot be cast to [[Ljava.lang.String;
... 33 elided
当我手动内联该调用以便没有类型参数化时,它可以正常工作:
scala> Array().foldLeft(Array(Array.empty[String])) {
| (acc: Array[Array[String]], s: String) => if (s.size > 1)
| acc :+ Array(s)
| else
| acc.init :+ (acc.last :+ s)
| }
res1: Array[Array[String]] = Array(Array())
知道这里发生了什么吗?我正在使用Scala 2.11.7。
最佳答案
看起来实例化嵌套数组会导致错误-即使在这么小的示例中,它也会崩溃:
def instantiate[A](ununsed_arg: Array[T])(implicit tag: ClassTag[A]) =
Array[Array[A]](Array.empty[A])
作为一种快速的解决方法,您可以使用ArrayBuilder创建一个数组:
def instantiate[T](a: Array[T])(implicit ctc: ClassTag[T]) = {
val builder = ArrayBuilder.make[Array[T]]
builder += Array.empty[T]
builder.result
}
还请注意,您的代码段直接将元素追加到数组中,这是个坏主意,因为它必须创建一个具有更多插槽的新数组并复制一个元素。看起来您应该使用
ArrayBuilder
或ArrayBuffer
进行处理,或者从列表中构建新的数据结构,最后将它们转换为Arrays
。