Come posso convertire immutable.Map
in mutable.Map
in Scala in modo da poter aggiornare i valori in Map
?
Come posso convertire immutable.Map
in mutable.Map
in Scala in modo da poter aggiornare i valori in Map
?
Risposte:
Il modo più pulito sarebbe usare la mutable.Map
fabbrica di varargs. A differenza ++
dell'approccio, questo utilizza il CanBuildFrom
meccanismo e quindi ha il potenziale per essere più efficiente se il codice della libreria è stato scritto per trarne vantaggio:
val m = collection.immutable.Map(1->"one",2->"Two")
val n = collection.mutable.Map(m.toSeq: _*)
Questo funziona perché a Map
può anche essere visto come una sequenza di coppie.
: _*
è molto simile all'ascrizione del tipo, che dice al compilatore esattamente quale tipo assegnare a una data espressione. Puoi immaginarlo qui come se dicesse "prendi questa sequenza e trattala come un numero di parametri vararg".
val myImmutableMap = collection.immutable.Map(1->"one",2->"two")
val myMutableMap = collection.mutable.Map() ++ myImmutableMap
O(1)
gradualmente. Questo sembra essere O(n)
, anche se ciò dipende ovviamente da quanto sia intelligente l'implementazione di ++
.
O(n)
. Nel limite quando cambi tutto, deve esserlo O(n)
, anche se potresti provare a differire la creazione della nuova copia per risparmiare tempo, oppure raddoppiare i tempi di accesso leggendo i changeset invece della mappa originale. Quale funziona meglio probabilmente dipende dal tuo caso d'uso.
Che ne dici di usare collection.breakOut?
import collection.{mutable, immutable, breakOut}
val myImmutableMap = immutable.Map(1->"one",2->"two")
val myMutableMap: mutable.Map[Int, String] = myImmutableMap.map(identity)(breakOut)
mutable.Map#apply
con un po 'più boilerplate.
A partire Scala 2.13
, tramite costruttori di fabbrica applicati con .to(factory)
:
Map(1 -> "a", 2 -> "b").to(collection.mutable.Map)
// collection.mutable.Map[Int,String] = HashMap(1 -> "a", 2 -> "b")
Esiste una variante per creare un mutabile vuoto Map
con valori predefiniti presi dall'immutabile Map
. Puoi memorizzare un valore e sovrascrivere il valore predefinito in qualsiasi momento:
scala> import collection.immutable.{Map => IMap}
//import collection.immutable.{Map=>IMap}
scala> import collection.mutable.HashMap
//import collection.mutable.HashMap
scala> val iMap = IMap(1 -> "one", 2 -> "two")
//iMap: scala.collection.immutable.Map[Int,java.lang.String] = Map((1,one), (2,two))
scala> val mMap = new HashMap[Int,String] {
| override def default(key: Int): String = iMap(key)
| }
//mMap: scala.collection.mutable.HashMap[Int,String] = Map()
scala> mMap(1)
//res0: String = one
scala> mMap(2)
//res1: String = two
scala> mMap(3)
//java.util.NoSuchElementException: key not found: 3
// at scala.collection.MapLike$class.default(MapLike.scala:223)
// at scala.collection.immutable.Map$Map2.default(Map.scala:110)
// at scala.collection.MapLike$class.apply(MapLike.scala:134)
// at scala.collection.immutable.Map$Map2.apply(Map.scala:110)
// at $anon$1.default(<console>:9)
// at $anon$1.default(<console>:8)
// at scala.collection.MapLike$class.apply(MapLike.scala:134)....
scala> mMap(2) = "three"
scala> mMap(2)
//res4: String = three
Avvertenza (vedi il commento di Rex Kerr): Non potrai rimuovere gli elementi provenienti dalla mappa immutabile:
scala> mMap.remove(1)
//res5: Option[String] = None
scala> mMap(1)
//res6: String = one