map

作者: 我傻笑你跑掉 | 来源:发表于2019-07-11 20:54 被阅读0次

    在其他语言诸如C++/Java中, map一般都以库的方式提供 如C++中的 std::map<> ,Java中的Hashmap<>,在golang中map为内置类型,不需要引用任何库.

    map是一堆未排序键值对的集合,map的key必须为可比较类型,比如 ==!=,map查找比线性查找快,但慢于索引查找(数组,切片)

    定义一个map

    格式 var name map[keytype]valuetype

    其中的keytype为map中键的类型,valuetype为map中值的类型,map还可以使用 := 或者 make()创建,如:

    var a map[int]int
    var b = map[string]string{}
    c := map[string]bool{}
    d := make(map[string]int)
    

    map的初始化

    map类似于数组和切片,可以在定义时直接指定初始值 如:

    var a = map[string]string{"name": "zhangsan", "age": "16", "sex": "男"}
    

    map元素访问

    上边定义并初始化了一个map,那么如何访问一个map中的元素呢?

    可以通过键(key)来访问指定的value 如:

    var a = map[string]string{"name": "zhangsan", "age": "16", "sex": "男"}
    fmt.Println(a["name"],a["age"])
    

    上述程序将输出: zhangsan 16

    将上述的代码修改一下

    var a = map[string]string{"name": "zhangsan", "age": "16", "sex": "男"}
    fmt.Println(a["name"], a["age"], a["parent"])
    

    map中并没有parent这个键,在诸如C++/Java或其他语言中,如果访问一个并不存在的键,将直接导致程序异常,所以判断一个map中是否存在某个键,就成了必需的步骤,但在golang中,这些都不是问题

    上述代码不会发生任何异常或者产生警告
    在golang中 如果访问一个未定义的key 将返回这个map中value的默认值,来验证一下

    package main
    
    import (
        "fmt"
    )
    
    func main() {
        var a = map[string]string{"name": "zhangsan", "age": "16", "sex": "男"}
        fmt.Println(a["parent"])
        if a["parent"] == "" {
            fmt.Println("yes")
        }
    
        var b = map[int]int{1: 1, 2: 2}
        fmt.Println(b[3])
        if b[3] == 0 {
            fmt.Println("yes")
        }
    }
    

    上述程序将输出
    //这个是空字符串
    yes
    0
    yes

    但显然我们并不想要这个结果,golang中提供了一种方式来判断数组中是否含有某个键

    格式 value,bool := map[key] 如下列程序

    func main() {
        var a = map[string]string{"name": "zhangsan", "age": "16", "sex": "男"}
        if value, ok := a["parent"]; ok == true {
            fmt.Println(value)
        } else {
            fmt.Println("key not in map")
        }
    }
    

    注意:golang中过的if语句中定义的变量在语句块外是访问不到的

    修改map中的元素

    和数组一样,可以使用key来修改map中的元素

    var a = map[string]string{"name": "zhangsan", "age": "16", "sex": "男"}
    fmt.Println(a)
    a["name"] = "李四"
    fmt.Println(a)
    

    上述代码将输出
    map[name:zhangsan age:16 sex:男]
    map[name:李四 age:16 sex:男]

    新增元素

    与slice不同,map元素的新增不需要使用copy()函数 可以直接 map[key] = value 的方式增加元素 如:

    package main
    
    import (
        "fmt"
    )
    
    func main() {
        var a = map[string]string{"name": "zhangsan", "age": "16", "sex": "男"}
        a["girlfriend"] = "苍老师"
        fmt.Println(a)
    }
    

    删除元素

    比slice相比,map提供delete()函数进行元素的删除
    格式 delete(map,key) 如下例:

    package main
    
    import (
        "fmt"
    )
    
    func main() {
        var a = map[string]string{"name": "zhangsan", "age": "16", "sex": "男"}
        delete(a, "name")
        fmt.Println(a)
    }
    

    以上程序将map[age:16 sex:男]

    map是引用类型

    map是引用类型,来看一个简单的例子

    func main() {
        var a = map[int]int{1: 1, 2: 2}
        b := a
        a[1] = 123456
        fmt.Println(a, b)
    }
    

    以上程序会输出 map[1:123456 2:2] map[1:123456 2:2]

    我们再传递进函数里面测试下,关于函数的定义后面我们会讲,此处只是一个demo

    func main() {
        var a = map[int]int{1: 1, 2: 2}
        b := a
        test(a)
        fmt.Println(a, b)
    }
    
    func test(a map[int]int) {
        a[1] = 111
    }
    

    以上程序会输出 map[2:2 1:111] map[1:111 2:2] 或者 map[1:111 2:2] map[1:111 2:2]

    为什么呢?仔细思考一下,这是否再一次说明了map是无序的呢?

    map的遍历

    与数组类似 map也可以使用 for range关键字来进行遍历

    var a = map[string]string{"name": "zhangsan", "age": "16", "sex": "男"}
    for key, value := range a {
        fmt.Println(key, value)
    }
    

    相关文章

      网友评论

        本文标题:map

        本文链接:https://www.haomeiwen.com/subject/nqejkctx.html