def flatArray[T : ClassManifest](a:Array[Any]) =
a.flatMap{
case ar:Array[T] => ar
case x: T => Array(x)
}
我試着也用#扁平化方法,但在NPE上失敗。
更新:要回答Jus12的問題:
def flatArray[T : Manifest](a:Array[Any]) =
a.flatMap{
case ar: Array[_] if ar.getClass.getComponentType == manifest[T].erasure => ar.asInstanceOf[Array[T]];
case x => Array(x.asInstanceOf[T])
}
整個球場的解決方案是不是類型安全的。原因是容納編譯器的類型推斷,推斷Array(Array(1, 2), 3, Array(4,5,6))
爲Array[Any]
。一個準確的類型是「一個Int
或Array[Int]
」的數組,但這是不可能的。什麼是是,每個元素都是Either[Int, Array[Int]]
和工作與創建要麼元素的數組:
object EitherView {
type ||[A, B] = Either[A, B]
// convenience of definition functions
private def l[A,B](a: A): ||[A,B] = Left(a)
private def r[A,B](b: B): ||[A,B] = Right(b)
// implicit defs - stuttering-or
implicit def aToOr2[A,B](a: A): A || B = l(a)
implicit def bToOr2[A,B](b: B): A || B = r(b)
implicit def aToOr3[A,B,C](a: A): A || B || C = l(l(a))
implicit def bToOr3[A,B,C](b: B): A || B || C = l(r(b))
implicit def aToOr4[A,B,C,D](a: A): A || B || C || D = l(l(l(a)))
implicit def bToOr4[A,B,C,D](b: B): A || B || C || D = l(l(r(b)))
implicit def aToOr5[A,B,C,D,E](a: A): A || B || C || D || E = l(l(l(l(a))))
implicit def bToOr5[A,B,C,D,E](b: B): A || B || C || D || E = l(l(l(r(b))))
// more? ...
}
import EitherView._
type CompoundArray[T] = Array[T || Array[T]]
object CompoundArray {
def apply[T](elems: (T || Array[T])*) = elems.toArray
}
def flatArray[T : Manifest](a:CompoundArray[T]) = {
a.flatMap{
case Left(x) => Array(x)
case Right(x) => x
}
}
參見:
scala> val a = CompoundArray[Int](Array(1, 2), 3, Array(4,5,6))
a: Array[EitherView.||[Int,Array[Int]]] = Array(Right([[email protected]), Left(3), Right([[email protected]))
scala> flatArray(a)
res0: Array[Int] = Array(1, 2, 3, 4, 5, 6)
scala> flatArray(CompoundArray[String](Array("hi"), "bye"))
res4: Array[String] = Array(hi, bye)
scala> flatArray(CompoundArray[String](Array("hi"), 3))
<console>:13: error: type mismatch;
found : Int(3)
required: EitherView.||[String,Array[String]]
flatArray(CompoundArray[String](Array("hi"), 3))
^
注:EitherView最初的想法是由@米奇Blevins:http://cleverlytitled.blogspot.com/2009/03/disjoint-bounded-views-redux.html
感謝您的解決方案。我得到了與上述的未經檢查的警告,所以我會去解決安瓦爾黎剎建議的解決方案。 – Jus12 2011-01-05 11:08:01
有人可以提出一種解決上述解決方案中的類型擦除警告的方法嗎?這是我得到的:「警告:類型模式中的抽象類型T Array [T]未被選中,因爲它被擦除消除 case ar:Array [T] => ar」 – Jus12 2011-01-05 23:37:45
感謝您的更新解決方案。我希望能有一個更簡單的解決方案。你能指出我對'EitherView'的原始參考嗎? – Jus12 2011-01-06 11:33:57