Map & Tuple

一、映射(Map)

1.1 构造Map

  1. // 初始化一个空 map
  2. val scores01 = new HashMap[String, Int]
  3. // 从指定的值初始化 Map(方式一)
  4. val scores02 = Map("hadoop" -> 10, "spark" -> 20, "storm" -> 30)
  5. // 从指定的值初始化 Map(方式二)
  6. val scores03 = Map(("hadoop", 10), ("spark", 20), ("storm", 30))

采用上面方式得到的都是不可变 Map(immutable map),想要得到可变 Map(mutable map),则需要使用:

  1. val scores04 = scala.collection.mutable.Map("hadoop" -> 10, "spark" -> 20, "storm" -> 30)

1.2 获取值

  1. object ScalaApp extends App {
  2. val scores = Map("hadoop" -> 10, "spark" -> 20, "storm" -> 30)
  3. // 1.获取指定 key 对应的值
  4. println(scores("hadoop"))
  5. // 2. 如果对应的值不存在则使用默认值
  6. println(scores.getOrElse("hadoop01", 100))
  7. }

1.3 新增/修改/删除值

可变 Map 允许进行新增、修改、删除等操作。

  1. object ScalaApp extends App {
  2. val scores = scala.collection.mutable.Map("hadoop" -> 10, "spark" -> 20, "storm" -> 30)
  3. // 1.如果 key 存在则更新
  4. scores("hadoop") = 100
  5. // 2.如果 key 不存在则新增
  6. scores("flink") = 40
  7. // 3.可以通过 += 来进行多个更新或新增操作
  8. scores += ("spark" -> 200, "hive" -> 50)
  9. // 4.可以通过 -= 来移除某个键和值
  10. scores -= "storm"
  11. for (elem <- scores) {println(elem)}
  12. }
  13. // 输出内容如下
  14. (spark,200)
  15. (hadoop,100)
  16. (flink,40)
  17. (hive,50)

不可变 Map 不允许进行新增、修改、删除等操作,但是允许由不可变 Map 产生新的 Map。

  1. object ScalaApp extends App {
  2. val scores = Map("hadoop" -> 10, "spark" -> 20, "storm" -> 30)
  3. val newScores = scores + ("spark" -> 200, "hive" -> 50)
  4. for (elem <- scores) {println(elem)}
  5. }
  6. // 输出内容如下
  7. (hadoop,10)
  8. (spark,200)
  9. (storm,30)
  10. (hive,50)

1.4 遍历Map

  1. object ScalaApp extends App {
  2. val scores = Map("hadoop" -> 10, "spark" -> 20, "storm" -> 30)
  3. // 1. 遍历键
  4. for (key <- scores.keys) { println(key) }
  5. // 2. 遍历值
  6. for (value <- scores.values) { println(value) }
  7. // 3. 遍历键值对
  8. for ((key, value) <- scores) { println(key + ":" + value) }
  9. }

1.5 yield关键字

可以使用 yield 关键字从现有 Map 产生新的 Map。

  1. object ScalaApp extends App {
  2. val scores = Map("hadoop" -> 10, "spark" -> 20, "storm" -> 30)
  3. // 1.将 scores 中所有的值扩大 10 倍
  4. val newScore = for ((key, value) <- scores) yield (key, value * 10)
  5. for (elem <- newScore) { println(elem) }
  6. // 2.将键和值互相调换
  7. val reversalScore: Map[Int, String] = for ((key, value) <- scores) yield (value, key)
  8. for (elem <- reversalScore) { println(elem) }
  9. }
  10. // 输出
  11. (hadoop,100)
  12. (spark,200)
  13. (storm,300)
  14. (10,hadoop)
  15. (20,spark)
  16. (30,storm)

1.6 其他Map结构

在使用 Map 时候,如果不指定,默认使用的是 HashMap,如果想要使用 TreeMap 或者 LinkedHashMap,则需要显式的指定。

  1. object ScalaApp extends App {
  2. // 1.使用 TreeMap,按照键的字典序进行排序
  3. val scores01 = scala.collection.mutable.TreeMap("B" -> 20, "A" -> 10, "C" -> 30)
  4. for (elem <- scores01) {println(elem)}
  5. // 2.使用 LinkedHashMap,按照键值对的插入顺序进行排序
  6. val scores02 = scala.collection.mutable.LinkedHashMap("B" -> 20, "A" -> 10, "C" -> 30)
  7. for (elem <- scores02) {println(elem)}
  8. }
  9. // 输出
  10. (A,10)
  11. (B,20)
  12. (C,30)
  13. (B,20)
  14. (A,10)
  15. (C,30)

1.7 可选方法

  1. object ScalaApp extends App {
  2. val scores = scala.collection.mutable.TreeMap("B" -> 20, "A" -> 10, "C" -> 30)
  3. // 1. 获取长度
  4. println(scores.size)
  5. // 2. 判断是否为空
  6. println(scores.isEmpty)
  7. // 3. 判断是否包含特定的 key
  8. println(scores.contains("A"))
  9. }

1.8 与Java互操作

  1. import java.util
  2. import scala.collection.{JavaConverters, mutable}
  3. object ScalaApp extends App {
  4. val scores = Map("hadoop" -> 10, "spark" -> 20, "storm" -> 30)
  5. // scala map 转 java map
  6. val javaMap: util.Map[String, Int] = JavaConverters.mapAsJavaMap(scores)
  7. // java map 转 scala map
  8. val scalaMap: mutable.Map[String, Int] = JavaConverters.mapAsScalaMap(javaMap)
  9. for (elem <- scalaMap) {println(elem)}
  10. }

二、元组(Tuple)

元组与数组类似,但是数组中所有的元素必须是同一种类型,而元组则可以包含不同类型的元素。

  1. scala> val tuple=(1,3.24f,"scala")
  2. tuple: (Int, Float, String) = (1,3.24,scala)

2.1 模式匹配

可以通过模式匹配来获取元组中的值并赋予对应的变量:

  1. scala> val (a,b,c)=tuple
  2. a: Int = 1
  3. b: Float = 3.24
  4. c: String = scala

如果某些位置不需要赋值,则可以使用下划线代替:

  1. scala> val (a,_,_)=tuple
  2. a: Int = 1

2.2 zip方法

  1. object ScalaApp extends App {
  2. val array01 = Array("hadoop", "spark", "storm")
  3. val array02 = Array(10, 20, 30)
  4. // 1.zip 方法得到的是多个 tuple 组成的数组
  5. val tuples: Array[(String, Int)] = array01.zip(array02)
  6. // 2.也可以在 zip 后调用 toMap 方法转换为 Map
  7. val map: Map[String, Int] = array01.zip(array02).toMap
  8. for (elem <- tuples) { println(elem) }
  9. for (elem <- map) {println(elem)}
  10. }
  11. // 输出
  12. (hadoop,10)
  13. (spark,20)
  14. (storm,30)
  15. (hadoop,10)
  16. (spark,20)
  17. (storm,30)

参考资料

  1. Martin Odersky . Scala 编程 (第 3 版)[M] . 电子工业出版社 . 2018-1-1
  2. 凯.S.霍斯特曼 . 快学 Scala(第 2 版)[M] . 电子工业出版社 . 2017-7