Go 映射(maps)

Go map(映射,下文不做翻译)就是在其它编程语言中众所周知的哈希表。map数据结构的主要优势就是其可以使用任意数据类型作为键值,但是对于Go map来说并不是所有的数据类型都能作为键值,只有可比较的类型才可以,意思是Go编译器能够区分不同的键值。或者简单来说,Go map的键值必须支持==操作符。显而易见,使用bool类型作为map的键值是非常不灵活的。另外,由于不同机器和操作系统的浮点数精度定义不同,使用浮点数作为键值可能会出现异常。

Go map的底层指向了一个哈希表!令人高兴的是Go已经隐藏了哈希表的实现及其复杂性,你将在第五章学习如何使用Go实现一个哈希表。

下面使用make()函数以string为键类型,以int作为值类型创建一个空的map:

iMap := make(map[string]int)

同样也可以使用map字面量创建并初始化一个map:

anotherMap := map[string]int {

“k1”: 12,

“k2”: 13

}

可以使用anotherMap[“k1”]可以获得对应的值,使用delete()删除一个键值对:

delete(anotherMap, “k1”)

遍历map中得元素可使用如下代码:

for key, value := range iMap {

fmt.Priintln(key,value)

}

usingMaps.go中的代码将会更加详细地展示map的用法。代码将会分为3部分,第一部分是:

  1. package main
  2. import (
  3. "fmt"
  4. )
  5. func main() {
  6. iMap := make(map[string]int)
  7. iMap["k1"] = 12
  8. iMap["k2"] = 13
  9. fmt.Println("iMap:", iMap)
  10. anotherMap := map[string]int {
  11. "k1": 12,
  12. "k2": 13,
  13. }

第二部分是:

  1. fmt.Println("anotherMap:",anotherMap)
  2. delete(anotherMap,"k1")
  3. delete(anotherMap, "k1")
  4. delete(anotherMap, "k1")
  5. fmt.Println("anotherMap:",anotherMap)
  6. _, ok := iMap["doseItExist"]
  7. if ok {
  8. fmt.Println("Exist!")
  9. } else {
  10. fmt.Println("dose NOT exist")
  11. }

这里你将学习到如何判断map中拥有某个键值对,这是很重要的知识点,如果不了解的话,你将无法判断一个map是否拥有你想要的信息。

需要注意的是,当你尝试使用一个并不存在的键去获取值的时候,返回值是0,但是你并不知道到底是某个键对应的值是0,还是由于所访问的键不存在而返回的0。

另外,代码中多次调用delete()去删除同一个元素并没有导致异常或者警告。

最后一部分代码展示了使用range关键字遍历map是非常简洁和方便的:

  1. for key, value := range iMap {
  2. fmt.Println(key, value)
  3. }
  4. }

执行usingMaps.go将会得到下面的输出:

$ go run usingMaps.go

iMap: map[k1:12 k2:13] anotherMap: map[k1:12 k2:13] anotherMap: map[k2:13] dose NOT exist k1 12 k2 13

你不能也不应该期望键值对是按顺序打印的,因为遍历map时其顺序是随机的。