Может быть, вы могли бы обернуть карту [Manifest, Any], гарантируя, что значения соответствуют ключам манифеста.
Возможный набросок этого. Сначала маленький помощник
class Typed[A](value: A)(implicit val key: Manifest[A]) {
def toPair: (Manifest[_], Any) = (key, value)
}
object Typed {
implicit def toTyped[A: Manifest](a: A) = new Typed(a)
implicit def toTypable[A](a: A) = new {
def typedAs[T >: A : Manifest] = new Typed[T](a)(manifest[T])
}
}
тогда сама оболочка (которая не является картой)
class TypedMap private(val inner: Map[Manifest[_], Any]) {
def +[A](t: Typed[A]) = new TypedMap(inner + t.toPair)
def +[A : Manifest](a: A) = new TypedMap(inner + (manifest[A] -> a))
def -[A : Manifest]() = new TypedMap(inner - manifest[A])
def apply[A : Manifest]: A = inner(manifest[A]).asInstanceOf[A]
def get[A : Manifest]: Option[A] = inner.get(manifest[A]).map(_.asInstanceOf[A])
override def toString = inner.toString
override def equals(other: Any) = other match {
case that: TypedMap => this.inner == that.inner
case _ => false
}
override def hashCode = inner.hashCode
}
object TypedMap {
val empty = new TypedMap(Map())
def apply(items: Typed[_]*) = new TypedMap(Map(items.map(_.toPair) : _*))
}
С этим вы можете сделать
import Typed._
val repository = TypedMap("foo", 12, "bar".typedAs[Any])
хранилище: TypedMap = Map (java.lang.String -> foo, Int -> 12, Any ->
бар)
Вы получаете элементы с
repository[String] // returns "foo"
repository.get[Any] // returns Some("bar")
Я думаю, что частный конструктор должен обеспечить безопасность _ asInstanceOf
. inner
может быть оставлено публичным, так как оно является неизменным. Таким образом, будет доступен богатый интерфейс Map
, но, к сожалению, не для создания другого TypedMap
.