亚洲在线久爱草,狠狠天天香蕉网,天天搞日日干久草,伊人亚洲日本欧美

為了賬號安全,請及時綁定郵箱和手機立即綁定
已解決430363個問題,去搜搜看,總會有你想問的

如何繞過Scala上的類型擦除?或者,為什么不能獲得集合的類型參數?

如何繞過Scala上的類型擦除?或者,為什么不能獲得集合的類型參數?

如何繞過Scala上的類型擦除?或者,為什么不能獲得集合的類型參數?在Scala上,一個可悲的事實是,如果您實例化一個List[Int],您可以驗證您的實例是一個列表,并且可以驗證它的任何單個元素是否是Int,但不能驗證它是否是List[Int],這一點很容易驗證:scala> List(1,2,3) match {      | case l : List[String] => println("A list of strings?!")      | case _ => println("Ok")      | }warning: there were unchecked warnings; re-run with -unchecked for details A list of strings?!未選中的選項將指責直接歸咎于類型擦除:scala>  List(1,2,3) match {      |  case l : List[String] => println("A list of strings?!")      |  case _ => println("Ok")      |  }<console>:6: warning: non variable type-argument String in type pattern is unchecked since it is eliminated by erasure              case l : List[String] => println("A list of strings?!")                  ^A list of strings?!為什么,我怎么才能避開它?
查看完整描述

3 回答

?
楊魅力

TA貢獻1811條經驗 獲得超6個贊

這個答案使用Manifest-API,它在Scala2.10中被廢棄。請參閱下面的答案,以獲得更多最新解決方案。

Scala是使用TypeErasure定義的,因為Java虛擬機(JVM)與Java不同,沒有泛型。這意味著,在運行時,只有類存在,而不是其類型參數。在本例中,JVM知道它正在處理scala.collection.immutable.List,但不是將此列表參數化為Int.

幸運的是,Scala中有一個特性可以讓您繞過它。是因為艙單..Manifest是一個類,其實例是表示類型的對象。由于這些實例是對象,所以可以傳遞它們、存儲它們,并通常對它們調用方法。在隱式參數的支持下,它成為一種非常強大的工具。例如,以下面的例子為例:

object Registry {
  import scala.reflect.Manifest

  private var map= Map.empty[Any,(Manifest[_], Any)] 

  def register[T](name: Any, item: T)(implicit m: Manifest[T]) {
    map = map.updated(name, m -> item)
  }

  def get[T](key:Any)(implicit m : Manifest[T]): Option[T] = {
    map get key flatMap {
      case (om, s) => if (om <:< m) Some(s.asInstanceOf[T]) else None
    }     
  }}scala> Registry.register("a", List(1,2,3))scala> Registry.get[List[Int]]("a")res6: Option[List[Int]] = Some(List(1, 2, 3))scala
  > Registry.get[List[String]]("a")res7: Option[List[String]] = None

在存儲元素時,我們也會存儲其中的“報表”。Manifest是一個類,其實例表示Scala類型。這些對象比JVM具有更多的信息,這使我們能夠測試完整的參數化類型。

不過,請注意,aManifest仍然是一個不斷發展的特征。作為其局限性的一個例子,它目前對方差一無所知,并且假設一切都是協變的。我希望一旦完成目前正在開發的Scala反射庫,它將變得更加穩定和可靠。


查看完整回答
反對 回復 2019-06-06
?
白豬掌柜的

TA貢獻1893條經驗 獲得超10個贊

您可以使用Typeable類型類無形為了得到你想要的結果,

REPL樣本會話,

scala> import shapeless.syntax.typeable._import shapeless.syntax.typeable._

scala> val l1 : Any = List(1,2,3)l1: Any = List(1, 2, 3)scala> l1.cast[List[String]]res0: Option[List[String]] = Nonescala
> l1.cast[List[Int]]res1: Option[List[Int]] = Some(List(1, 2, 3))

這個cast在此范圍內,操作將盡可能精確地擦除wrt。Typeable實例可用。


查看完整回答
反對 回復 2019-06-06
  • 3 回答
  • 0 關注
  • 1130 瀏覽

添加回答

舉報

0/150
提交
取消
微信客服

購課補貼
聯系客服咨詢優惠詳情

幫助反饋 APP下載

慕課網APP
您的移動學習伙伴

公眾號

掃描二維碼
關注慕課網微信公眾號