web-dev-qa-db-ja.com

Scalaでデフォルトの操作でマップを実装する方法

class DefaultListMap[A, B <: List[B]] extends HashMap[A, B] {
    override def default(key: A) = List[B]() 
  }

地図を作成したくないA -> List[B]。私の場合はLong -> List[String]しかし、値のないマップからキーを取得すると、Listがスローされるのではなく、空のExceptionを作成したいと思います。さまざまな組み合わせを試しましたが、上記のコードをコンパイラに渡す方法がわかりません。

前もって感謝します。

45
Lukasz

なぜwithDefaultValue(value)を使用しないのですか?

scala> val m = Map[Int, List[String]]().withDefaultValue(List())
m: scala.collection.immutable.Map[Int,List[String]] = Map()

scala> m(123)
res1: List[String] = List()
106

applyを使用してマップにアクセスするのではなく、常にgetを使用して、Option[V]を返し、次にgetOrElseを返すことができます。

map.get(k) getOrElse Nil

scalaz関数型プログラミングライブラリの優れた機能の1つは、単項演算子~で、これは「またはゼロ」を意味します"、値の型に"ゼロ "が定義されている限り(もちろんListが行い、ゼロはNilです)。したがって、コードは次のようになります。

~map.get(k)

(たとえば)値がIntDoubleなど(Zeroタイプクラスがあるもの)の場合に同じ構文が機能するため、これは二重に役立ちます。


特にisDefinedAtメソッドに関してこれがどのように動作するかにより、scalaメーリングリストではMap.withDefaultの使用について多くの議論がありました。私はこの理由でそれを避けようとする傾向があります。

23
oxbow_lakes

withDefaultValueにメソッドMapがあります:

scala> val myMap = Map(1 -> List(10), 2 -> List(20, 200)).withDefaultValue(Nil)
myMap: scala.collection.immutable.Map[Int,List[Int]] = Map((1,List(10)), (2,List(20, 200)))

scala> myMap(2)
res0: List[Int] = List(20, 200)

scala> myMap(3)
res1: List[Int] = List()
11
kassens

このためのメソッドがすでにあるのに、なぜマップを操作したいのですか?

val m = Map(1L->List("a","b"), 3L->List("x","y","z"))  
println(m.getOrElse(1L, List("c"))) //--> List(a, b)
println(m.getOrElse(2L, List("y"))) //--> List(y)
3
Landei

withDefaultも使用できます。

/** The same map with a given default function.
 *  Note: `get`, `contains`, `iterator`, `keys`, etc are not affected
 *  by `withDefault`.
 *
 *  Invoking transformer methods (e.g. `map`) will not preserve the default value.
 *
 *  @param d     the function mapping keys to values, used for non-present keys
 *  @return      a wrapper of the map with a default value
 */
 def withDefault[B1 >: B](d: A => B1): immutable.Map[A, B1]

例:

scala> def intToString(i: Int) = s"Integer $i"
intToString: (i: Int)String

scala> val x = Map[Int, String]().withDefault(intToString)
x: scala.collection.immutable.Map[Int,String] = Map()

scala> x(1)
res5: String = Integer 1

scala> x(2)
res6: String = Integer 2

お役に立てれば。

2
tharindu_DG