1

我有一个 type 的地图Map[_, Any],我想以它们的本机格式提取值(不求助于.asInstanceOf[_])。

像这样的东西...

val m: Map[String, Any] = Map("i" -> 1, "s" -> "test")

val i: Option[Int] = m.get("i")
val s: Option[String] = m.get("s")

显然那是失败的。

我不喜欢这种方法,但我想我可以做这样的事情......但即使这样仍然出现Any而不是Intor String

trait MyType[A] {
  def value: A
}

implicit class MyInt(i: Int) extends MyType[Int] { def value: Int = i }
implicit class MyString(s: String) extends MyType[String] { def value: String = s }

val m: Map[String, MyType[_]] = Map("i" -> 1, "s" -> "test")

val i: Option[Int] = m.get("i").map(_.value)
val s: Option[String] = m.get("s").map(_.value)

然后我想也许是一些包装Map...

case class MyMap(m: Map[String, Any]) {
  def get[A](k: String)(implicit ev: Option[Any] => Option[A]): Option[A] = m.get(k)
}

但这仍然是Any. 我只是不知道如何转换 Any => native。

所以我的问题是...

  1. 为什么会失败?
  2. 以原始格式获取值的更好方法是什么?最简单和/或没有外部依赖是理想的......但我真的对任何事情都持开放态度(尽管需要注意的是我现在仍在使用 scala 2.11)。

谢谢!

4

1 回答 1

4

由于评论中已经解释过的原因,您无法猜测运行时类型 - 此信息不存在,一旦存在Any,所有类型信息都会丢失,您无能为力。

因此,您必须自己提供预期的类型。.as[T]辅助方法怎么样?

// This code is specifically for 2.11, please don't use it for more recent versions,
// see link below.

val m: Map[String, Any] = Map("i" -> 1, "s" -> "test")

import scala.reflect.{ ClassTag, classTag }

implicit class As(a: Any) {
  def as[T](implicit ct: ClassTag[T]): Option[T] = ct.unapply(a)
}

println(m("i").as[Int].map(_ + 41).get)
println(m("s").as[String].map("This is a " + _).get)

这将打印

42
This is a test

简要说明:

  • 包装器“As拉皮条”所有对象,并将方法附加.as到所有内容。
  • 执行unapply检查、转换和包装在一个Option.

它不适用于List[Int]vs.List[String]等泛型类型,因为此信息在运行时根本不可用。

编辑:感谢@MarioGalic 大大简化了解决方案。

于 2021-05-10T21:33:13.383 回答