映射是一种数据结构,用于存储一系列无序的键值对,它基于键来存储值。映射的特点是能够基于键快速检索数据。键就像是数组的索引一样,指向与键关联的值。
与 C++、Java 等编程语言不同,在 Golang 中使用映射不需要引入任何库。因此 Golang 的映射使用起来更加方便。我们可以通过下图简要的理解一下映射中键值对的关系:

图中的每个键值对表示一种颜色的字符串名称及其对应的十六进制值,其中名称为键,十六进制数为值。

映射的实现

映射是一个数据集合,所以可以是使用类似处理数组和切片的方式来迭代映射中的元素。但映射是无序集合,所以即使以同样的顺序保存键值对,迭代映射时,元素的顺序可能会不一样。无序的原因是映射的实现使用了哈希表。
Golang 中的映射在底层是用哈希表实现的,在 /usr/local/go/src/runtime/hashmap.go 中可以查看它的实现细节。而 C++ 中的映射则是使用红黑树实现的。

创建和初始化映射

Golang 中有很多种方法可以创建并初始化映射,可以使用内置的 make 函数,也可以使用映射字面量。

使用 make 函数声明映射

// 创建一个映射,键的类型是 string,值的类型是 int
myMap := make(map[string]int)

使用字面量声明映射

// 创建一个映射,键和值的类型都是 string
// 使用两个键值对初始化映射
myMap := map[string]string{"Red": "#da1337", "Orange": "#e95a22"}

创建映射时,更常用的方法是使用映射字面量。映射的初始长度会根据初始化时指定的键值对的数量来确定。

映射的键可以是任何值
这个值的类型可以是内置的类型,也可以是结构类型,只要这个值可以使用 == 运算符做比较。切片、函数以及包含切片的结构类型,这些类型由于具有引用语义,不能作为映射的键,使用这些类型会造成编译错误:

// 使用映射字面量声明空映射
// 创建一个映射,使用字符串切片作为映射的键
myMap := map[[]string]int{}

如果你使用的 IDE 支持语法检查,就会提示这段代码有语法错误:

如果直接编译上面的代码,会得到一个编译时错误:
invalid map key type []string

虽然切片不能作为映射的键,但是却可以作为映射的值,这个在使用一个映射键对应一组数据时,会非常有用:

// 声明一个存储字符串切片的映射
// 创建一个映射,使用字符串切片作为值
myMap := map[int][]string{}

元素赋值

通过指定适当类型的键并给这个键赋一个值就完成了映射的键值对赋值:

// 创建一个空映射,用来存储颜色以及颜色对应的十六进制代码
myColors := map[string]string{}
// 将 Red 的代码加入到映射
myColors["Red"] = "#da1337"

与切片类似,可以通过声明一个未初始化的映射来创建一个值为 nil 的映射(一般称为 nil 映射),nil 映射不能用于存储键值对:

// 通过声明映射创建一个 nil 映射
var myColors map[string]string
// 将 Red 的代码加入到映射
myColors["Red"] = "#da1337"

运行这段代码会产生一个运行时错误:
panic: assignment to entry in nil map

查找与遍历

测试键值是否存在
查找映射里是否存在某个键是映射的一个基本操作。这个操作往往需要用户写一些逻辑代码,根据逻辑代码确定是否完成了某个操作或者映射里是否缓存了某些数据。查找操作也可以用来比较两个映射,确定哪些键值对互相匹配,哪些键值对不匹配。
有两种方法可以检查键值对是否存在,第一种方式是获取键值对中的值以及一个表示这个键是否存在的布尔类型标志:

// 获取键 Blue 对应的值
value, exists := myColors["Blue"]
// 这个键存在吗?
if exists {fmt.Println(value)
}

另一中方式是,只返回键对应的值,然后通过判断这个值是不是零值来确定键是否存在:

// 获取键 Blue 对应的值
value := myColors["Blue"]
// 这个键存在吗?
if value != "" {fmt.Println(value)
}

显然,这种方式只能用在映射存储的值都是非零值的情况下。
注意:在 Golang 中,通过键来索引映射时,即便这个键不存在也总会返回一个值。在这种情况下,返回的是该值对应的类型的零值。

遍历映射
和遍历数组、切片一样,使用关键字 range 可以遍历映射中的所有值。但对映射来说,range 返回的不是索引和值,而是键值对:

// 创建一个映射,存储颜色以及颜色对应的十六进制代码
myColors := map[string]string{"AliceBlue":"#f0f8ff","Coral":"#ff7F50","DarkGray":"#a9a9a9","ForestGreen": "#228b22",
}
// 显示映射里的所有颜色
for key, value := range myColors {fmt.Printf("Key: %s  Value: %s\n", key, value)
}

执行上面的代码,输出如下:

Key: AliceBlue  Value: #f0f8ff
Key: Coral  Value: #ff7F50
Key: DarkGray  Value: #a9a9a9
Key: ForestGreen  Value: #228b22

删除映射中的元素

Golang 提供了一个内置的函数 delete() 用于删除集合中的元素,下面是一个简单的例子:
delete(myMap, "hello")
上面的代码将从 myMap 中删除键为 hello 的键值对。如果 hello 这个键不存在,那么这个调用将什么都不会发生,也不会有什么副作用。但是如果传入的映射的变量的值为 nil,该调用将导致程序抛出异常(panic)。
还以前面定义的 myColors 映射为例,我们用 delete() 函数删除其中的 Coral:

// 创建一个映射,存储颜色以及颜色对应的十六进制代码
myColors := map[string]string{"AliceBlue":"#f0f8ff","Coral":"#ff7F50","DarkGray":"#a9a9a9","ForestGreen": "#228b22",
}
// 删除键为Coral的键值对
delete(myColors, "Coral")// 显示映射里的所有颜色
for key, value := range myColors {fmt.Printf("Key: %s  Value: %s\n", key, value)
}

执行上面的代码,发现输出的结果中已经没有 Coral 了:

Key: DarkGray  Value: #a9a9a9
Key: ForestGreen  Value: #228b22
Key: AliceBlue  Value: #f0f8ff

在函数间传递映射

在函数间传递映射并不会制造出该映射的一个副本。实际上,当传递映射给一个函数,并对这个映射做了修改时,所有对这个映射的引用都会察觉到这个修改:

package main
import "fmt"func main() {// 创建一个映射,存储颜色以及颜色对应的十六进制代码myColors := map[string]string{"AliceBlue":"#f0f8ff","Coral":"#ff7F50","DarkGray":"#a9a9a9","ForestGreen": "#228b22",}// 显示映射里的所有颜色for key, value := range myColors {fmt.Printf("Key: %s  Value: %s\n", key, value)}fmt.Println()// 调用函数来移除指定的键removeColor(myColors, "Coral")// 显示映射里的所有颜色for key, value := range myColors {fmt.Printf("Key: %s Value: %s\n", key, value)}
}// removeColor 将指定映射里的键删除
func removeColor(colors map[string]string, key string) {delete(colors, key)
}

运行上面的程序,输出如下结果:

Key: Coral  Value: #ff7F50
Key: DarkGray  Value: #a9a9a9
Key: ForestGreen  Value: #228b22
Key: AliceBlue  Value: #f0f8ffKey: AliceBlue Value: #f0f8ff
Key: DarkGray Value: #a9a9a9
Key: ForestGreen Value: #228b22

可以看到,在调用了 removeColor 函数后,main 函数中引用的映射中也不再有 Coral 颜色了。这个特性和切片类似,保证可以用很小的成本来复制映射。

总结

映射是 Golang 中内置的保存键值对类型数据的类型。从本文的示例中可以看出,Golang 映射实现的简单易用,并且在函数间传递时的性能很好、开销很低。

golang 映射 map 简介相关推荐

  1. Golang sync.Map 简介与用法

    Golang 中的 map 在并发情况下,只读是线程安全的,并发读写线程不安全.为了解决这个问题,Golang 提供了语言层级的并发读写安全的 sync.Map. type Map struct {/ ...

  2. [译] part 13: golang 映射 map

    原文地址:Part 13: Maps 原文作者:Naveen R 译者:咔叽咔叽 转载请注明出处. 什么是 map map 是 Go 中的内置类型,它将值与键相关联.可以使用相应的键查找该值. 怎么创 ...

  3. Power BI for Office 365(六)Power Map简介

    Power BI for Office 365(六)Power Map简介 本文是转载: 来自  http://www.cnblogs.com/aspnetx/p/3290085.html#undef ...

  4. [UVA156]反片语 Ananagrams 题解(映射:map 详解)

    反片语 Ananagrams - 洛谷 Most crossword puzzle fans are used to anagrams - groups of words with the same ...

  5. Java中的映射Map - 入门篇

    前言 大家好啊,我是汤圆,今天给大家带来的是<Java中的映射Map - 入门篇>,希望对大家有帮助,谢谢 简介 前面介绍了集合List,这里开始简单介绍下映射Map,相关类如下图所示 正 ...

  6. 2021年大数据常用语言Scala(二十二):函数式编程 映射 map

    目录 映射 | map 用法 案例一 案例二 映射  map 集合的映射操作是将来在编写Spark/Flink用得最多的操作,是我们必须要掌握的.因为进行数据计算的时候,就是一个将一种数据类型转换为另 ...

  7. STL map 简介

    STL map 简介 转载于:http://www.cnblogs.com/TianFang/archive/2006/12/30/607859.html 1.目录 map简介 map的功能 使用ma ...

  8. 元组Tuple、数组Array、映射Map

    一.元组Tuple 元组Tuple是不同类型的值的聚集,元组的值将单个的值包含在圆括号中来构成,元组可以包含一个不同类型的元素 如 val riple = (100, "Scala" ...

  9. golang 中 map 转 struct

    golang 中 map 转 struct package mainimport ("fmt""github.com/goinggo/mapstructure" ...

最新文章

  1. 植保口的面上项目共153项,系统总结
  2. leecode11 盛水最多的容器
  3. python idle打不开_孩子,给你的Python安个家吧!
  4. linux drbd同步,DRBD数据镜像主备节点同步数据
  5. STM32使用PWM输入模式测试频率和占空比
  6. 理解Android的手势识别
  7. 初学Reporting Service2008
  8. python矩阵转置_Python 矩阵转置的几种方法小结
  9. [高光谱] Hyperspectral-Classification Pytorch 的高光谱场景的通用类 HyperX
  10. top命令详解(转载)
  11. UOJ#52. 【UR #4】元旦激光炮(交互)
  12. ROS@Ubuntu16.04体验记录
  13. 股票大宗交易对股价走势的影响
  14. 云桌面服务器+搭建,搭建自己的云桌面服务器
  15. Android BLE操作成功或失败status code对应解释
  16. 【Python】模块(Module)、包(Package)以及相对导入(relative import)和绝对导入(absolute import)
  17. 计算机图标下面有颜色,小编教你电脑桌面图标有蓝色阴影怎么去掉
  18. 抛物型偏微分方程的Crank-Nicolson 方法; Richardson 外推法;紧差分法
  19. php中Mysql常用操作类收集
  20. 论遇到事情的沉着与冷静

热门文章

  1. Python Module_sys/random
  2. Altium Designer原理图和PCB中对元件垂直、水平镜像翻转
  3. 【详细】Android入门到放弃篇-YES OR NO-》各种UI组件,布局管理器,单元Activity
  4. springMVC通过ajax传递参数list对象或传递数组对象到后台
  5. Haproxy代理配置---传输层
  6. Delphi 中的 XMLDocument 类详解(5) - 获取元素内容
  7. oracle空间管理
  8. office excel Query 功能
  9. shell脚本分析mysql慢查询日志(slow log)
  10. IdentityServer4 使用OpenID Connect添加用户身份验证