5

docs.scala-lang.org上阅读HashTrieMaps上的这句话后:

例如,要在地图中找到给定的键,首先要获取该键的哈希码。然后,哈希码的最低 5 位用于选择第一个子树,然后是接下来的 5 位,以此类推。一旦存储在节点中的所有元素的哈希码在此级别之前选择的位中彼此不同,选择就会停止。

我认为这是一个很棒的(阅读:快!)集合来存储我的 Map[String, Long]。

在我的 Play Framework(使用 Scala)中,我使用 Anorm 加载了大约 18k 个元素的这段代码。加载需要几秒钟(没什么大不了的,但有什么提示吗?)。我想把它放在“内存中”,以便快速查找字符串到长翻译。

val data = DB.withConnection { implicit c ⇒
  SQL( "SELECT stringType, longType FROM table ORDER BY stringType;" )
    .as( get[String]( "stringType" ) ~ get[Long]( "longType " )
    map { case ( s ~ l ) ⇒ s -> l }* ).toMap.withDefaultValue( -1L )
}

此代码data的类型为class scala.collection.immutable.Map$WithDefault. 我希望这是一种类型HashTrieMap(或者HashMap,据我所知,链接引用的所有 Scala HashMap 都属于 HashTrieMap?)。奇怪的是,我找不到如何将其转换为 HashTrieMap 的方法。(我是 Scala、Play 和 Anorm 的新手。)

// Test for the repl (2.9.1.final). Map[String, Long]:
val data = Map( "Hello" -> 1L, "world" -> 2L ).withDefaultValue ( -1L )
data: scala.collection.immutable.Map[java.lang.String,Long] =
  Map(Hello -> 1, world -> 2)

// Google showed me this, but it is still a Map[String, Long].
val hm = scala.collection.immutable.HashMap( data.toArray: _* ).withDefaultValue( -1L )

// This generates an error.
val htm = scala.collection.immutable.HashTrieMap( data.toArray: _* ).withDefaultValue( -1L )

所以我的问题是如何将 MapWithDefault 转换为 HashTrieMap(或 HashMap,如果它共享 HashTrieMap 的实现)?

欢迎任何反馈。

4

1 回答 1

9

正如您指出的文档所解释的那样,不可变映射已经在后台实现为HashTrieMaps. 您可以在 REPL 中轻松验证这一点:

scala> println( Map(1->"one", 2->"two", 3->"three", 4->"four", 5->"five").getClass )
class scala.collection.immutable.HashMap$HashTrieMap

所以你没有什么特别的事情要做,你的代码已经在HashMap.HashTrieMap你甚至没有意识到的情况下使用。

immutable.Map 更准确地说,是的默认实现immutable.HashMap,由 进一步细化(扩展)immutable.HashMap.HashTrieMap。请注意,虽然小的不可变映射不是的实例immutable.HashMap.HashTrieMap,但作为特殊情况实现(这是一种优化)。有一个特定的大小阈值,它们开始被执行为 immutable.HashMap.HashTrieMap. 例如,在 REPL 中输入以下内容:

val m0 = HashMap[Int,String]()
val m1 = m0 + (1 -> "one")
val m2 = m1 + (2 -> "two")
val m3 = m2 + (3 -> "three")
println(s"m0: ${m0.getClass.getSimpleName}, m1: ${m1.getClass.getSimpleName}, m2: ${m2.getClass.getSimpleName}, m3: ${m3.getClass.getSimpleName}")

将打印:

m0: EmptyHashMap$, m1: HashMap1, m2: HashTrieMap, m3: HashTrieMap

所以这里的空地图是EmptyHashMap$. 添加一个元素得到 a HashMap1,再添加一个元素最终得到 a HashTrieMap


最后,使用withDefaultValue不会改变任何东西,因为withDefaultValue只会返回一个Map.WithDefault包含初始映射的实例(仍然是 a HashMap.HashTrieMap)。

于 2013-02-04T13:49:56.173 回答