idiomatic “get or else update” for immutable.Map?

后端 未结 4 1518
再見小時候
再見小時候 2021-02-04 05:01

What is the idiomatic way of a getOrElseUpdate for immutable.Map instances?. I use the snippet below, but it seems verbose and inefficient

var map = Map[Key, Val         


        
相关标签:
4条回答
  • 2021-02-04 05:33

    Why not use withDefault or withDefaultValue if you have an immutable map?

    0 讨论(0)
  • 2021-02-04 05:41

    Let me summarise your problem:

    • You want to call a method on a immutable data structure
    • You want it to return some value and reassign a var
    • Because the data structure is immutable, you’ll need to
      • return a new immutable data structure, or
      • do the assignment inside the method, using a supplied closure

    So, either your signature has to look like

    def getOrElseUpdate(key: K): Tuple2[V, Map[K,V]]
    //... use it like
    val (v, m2) = getOrElseUpdate(k)
    map = m2
    

    or

    def getOrElseUpdate(key: K, setter: (Map[K,V]) => Unit): V
    //... use it like
    val v = getOrElseUpdate(k, map = _)
    

    If you can live with one of these solutions, you could add your own version with an implicit conversion but judging by the signatures alone, i wouldn’t think any of these is in the standard library.

    0 讨论(0)
  • 2021-02-04 05:46

    There's no such way - map mutation (update), when you're getting a map value, is a side effect (which contradicts to immutability/functional style of programming).

    When you want to make a new immutable map with the default value, if another value for the specified key doesn't exist, you can do the following:

    map + (key -> map.getOrElse(key, new Value)) 
    
    0 讨论(0)
  • 2021-02-04 05:51

    I would probably implement a getOrElseUpdated method like this:

    def getOrElseUpdated[K, V](m: Map[K, V], key: K, op: => V): (Map[K, V], V) =
      m.get(key) match {
        case Some(value) => (m, value)
        case None => val newval = op; (m.updated(key, newval), newval)
      }
    

    which either returns the original map if m has a mapping for key or another map with the mapping key -> op added. The definition of this method is similar to getOrElseUpdate of mutable.Map.

    0 讨论(0)
提交回复
热议问题