创建map
var m1 map[int]string //只是声明一个map,没有初始化, 为空(nil)map
fmt.Println(m1 == nil) //true
//m1[1] = “Luffy” //nil的map不能使用err, panic: assignment to entry in nil map
m2 := map[int]string{} //m2, m3的创建方法是等价的
m3 := make(map[int]string)
fmt.Println(m2, m3) //map[] map[]
m4 := make(map[int]string, 10) //第2个参数指定容量
fmt.Println(m4) //map[]
创建m4的方法指定了map的初始创建容量。 与slice类似,后期在使用过程中,map可以自动扩容。只不过map更方便一些,不用借助类似append的函数,直接赋值即可。如,m1[17] = “Nami”。赋值过程中,key如果与已有map中key重复,会将原有map中key对应的value覆盖。
但是!对于map而言,可以使用len()函数,但不能使用cap()函数。
初始化map
也可以直接指定初值,要保证key不重复。
//1、定义同时初始化
var m1 map[int]string = map[int]string{1: “Luffy”, 2: “Sanji”}
fmt.Println(m1) //map[1:Luffy 2:Sanji]
//2、自动推导类型 :=
m2 := map[int]string{1: “Luffy”, 2: “Sanji”}
fmt.Println(m2)
常用操作
赋值
m1 := map[int]string{1: “Luffy”, 2: “Sanji”}
m1[1] = “Nami” //修改
m1[3] = “Zoro” //追加, go底层会自动为map分配空间
fmt.Println(m1) //map[1:Nami 2:Sanji 3:Zoro]
m2 := make(map[int]string, 10) //创建map
m2[0] = “aaa”
m2[1] = “bbb”
fmt.Println(m2) //map[0:aaa 1:bbb]
fmt.Println(m2[0], m2[1]) //aaa bbb
遍历
Map的迭代顺序是不确定的,并且不同的哈希函数实现可能导致不同的遍历顺序。在实践中,遍历的顺序是随机的,每一次遍历的顺序都不相同。这是故意的,每次都使用随机的遍历顺序可以强制要求程序不会依赖具体的哈希函数实现。
m1 := map[int]string{1: “Luffy”, 2: “Sanji”}
//遍历1,第一个返回值是key,第二个返回值是value
for k, v := range m1 {
fmt.Printf(“%d —-> %s\n”, k, v)
//1 —-> Luffy
//2 —-> yoyo
}
//遍历2,第一个返回值是key,第二个返回值是value(可省略)
for k := range m1 {
fmt.Printf(“%d —-> %s\n”, k, m1[k])
//1 —-> Luffy
//2 —-> Sanji
}
有时候可能需要知道对应的元素是否真的是在map之中。可以使用下标语法判断某个key是否存在。map的下标语法将产生两个值,其中第二个是一个布尔值,用于报告元素是否真的存在。
如果key存在,第一个返回值返回value的值。第二个返回值为 true。
value, ok := m1[1]
fmt.Println(“value = “, value, “, ok = “, ok) //value = mike , ok = true
如果key不存在,第一个返回值为空,第二个返回值为false。
value2, has := m1[3]
fmt.Println(“value2 = “, value2, “, has = “, has) //value2 = , has = false
删除
使用delete()函数,指定key值可以方便的删除map中的k-v映射。
m1 := map[int]string{1: “Luffy”, 2: “Sanji”, 3: “Zoro”}
for k, v := range m1 { //遍历,第一个返回值是key,第二个返回值是value
fmt.Printf(“%d —-> %s\n”, k, v)
}
//1 —-> Sanji
//2 —-> Sanji
//3 —-> Zoro
delete(m1, 2) //删除key值为2的map
for k, v := range m1 {
fmt.Printf(“%d —-> %s\n”, k, v)
}
//1 —-> Luffy
//3 —-> Zoro
delete()操作是安全的,即使元素不在map中也没有关系;如果查找删除失败将返回value类型对应的零值。
如:
delete(m1, 5) //删除key值为5的map
for k, v := range m1 {
fmt.Printf(“%d —-> %s\n”, k, v)
}
//1 —-> Luffy
//3 —-> Zoro
map输出结果依然是原来的样子,且不会有任何错误提示。