从浅入深带你掌握Golang数据结构map
作者:金刀大菜牙 发布时间:2023-06-21 16:42:43
在 Go 语言中,map 是一种非常常见的数据类型,它可以用于快速地检索数据。Go 语言中的 map 与其他编程语言中的类似的数据类型相比,具有一些独特的特点,使其更加高效和灵活。本篇文章将介绍 Go 语言中的 map,包括 map 的定义、初始化、操作和优化。
1. 什么是 map
map 是一种 key-value 结构的数据类型,key 是唯一的,value 可以重复。在 Go 语言中,map 的定义格式为:
map[keyType]valueType
其中,keyType 和 valueType 分别表示 map 的 key 和 value 的数据类型。例如:
var m map[string]int
表示定义了一个 key 类型为 string,value 类型为 int 的 map 变量 m。
2. map 的初始化
Go 语言中的 map 需要通过 make 函数来初始化,如下所示:
m := make(map[keyType]valueType)
其中,keyType 和 valueType 分别表示 map 的 key 和 value 的数据类型。例如:
m := make(map[string]int)
表示定义了一个 key 类型为 string,value 类型为 int 的 map 变量 m。如果希望 map 有一些默认值,可以使用字面量来初始化 map,例如:
m := map[string]int{"foo": 1, "bar": 2}
这样,就定义了一个初始值包含 "foo": 1 和 "bar": 2 的 map 变量 m。
3. map 的操作
Go 语言中的 map 提供了一系列的操作函数,可以方便地进行添加、删除、查找等操作。
3.1 添加和修改元素
要向 map 中添加一个元素,可以使用下面的语法:
m[key] = value
如果 key 已经存在,那么 value 会被覆盖。例如:
m := make(map[string]int)
m["foo"] = 1
m["bar"] = 2
m["foo"] = 3 // 覆盖已有的"foo": 1
3.2 删除元素
要删除 map 中的一个元素,可以使用下面的语法:
delete(m, key)
其中,m 表示要删除元素的 map 变量,key 表示要删除的元素的 key。例如:
m := make(map[string]int)
m["foo"] = 1
delete(m, "foo") // 删除"foo": 1
3.3 查找元素
要查找 map 中的一个元素,可以使用下面的语法:
value, ok := m[key]
其中,m 表示要查找元素的 map 变量,key 表示要查找的元素的 key,value 表示查找到的元素的值,ok 表示是否查找成功。例如:
m := make(map[string]int)
m["foo"] = 1
value, ok := m["foo"]
if ok {
fmt.Println(value) // 输出1
}
3.4 遍历元素
要遍历 map 中的所有元素,可以使用 for...range 循环语句,例如:
m := map[string]int{"foo": 1, "bar": 2}
for key, value := range m {
fmt.Println(key, value)
}
这样就会遍历输出所有的 key 和 value。
4. map 的优化
在使用 map 时,需要注意一些性能优化的技巧,以提高 map 的性能。
4.1 预分配 map 的大小
在使用 map 时,如果已经知道 map 的大小,可以预先分配 map 的大小,以减少 map 扩容的次数,从而提高性能。例如:
m := make(map[string]int, 1000)
这样就预分配了 map 的大小为 1000。
4.2 使用值类型作为 map 的 key
在使用 map 时,如果使用指针类型或复杂类型作为 key,会导致 map 的性能下降。因此,最好使用值类型作为map 的 key。例如:
type myStruct struct {
foo int
bar string
}
m := make(map[myStruct]int)
这样就使用了值类型 myStruct 作为 map 的 key。
4.3 不要在循环中使用值类型作为 map 的 key
在使用 map 时,如果在循环中使用值类型作为 map 的 key,会导致每次循环都要复制一份值类型,从而降低性能。因此,最好在循环中使用指针类型作为 map 的 key。例如:
m := make(map[*myStruct]int)
for i := 0; i < 1000; i++ {
key := &myStruct{foo: i, bar: "test"}
m[key] = i
}
这样就使用了指针类型 *myStruct 作为 map 的 key。
4.4 使用 sync.Map 代替 map
在并发环境中,如果使用 map,需要对 map 进行加锁保证并发安全。但是,加锁会导致性能下降。因此,最好使用 sync.Map 代替 map。sync.Map 是 Go 语言中的一个并发安全的 map 实现。
使用 sync.Map 的方式与使用普通 map 基本相同,只需要将 map 类型替换为 sync.Map 即可。例如,可以使用以下方式创建一个sync.Map:
var m sync.Map
然后可以通过以下方式向 sync.Map 中添加元素:
m.Store("key", "value")
通过以下方式从 sync.Map 中读取元素:
value, ok := m.Load("key")
需要注意的是,sync.Map 虽然是一种线程安全的 map 实现,但是在高并发场景下仍然会存在一些性能问题,因为它需要进行额外的并发安全控制。因此,在不需要并发安全的场合,可以使用普通的 map 来提高性能。
5. 总结
map 是 Golang 中非常有用的一种数据结构,可以用来存储键值对的集合。在使用 map 时,需要注意一些性能优化的技巧,如预分配 map 的大小、使用值类型作为 map 的键、避免在循环中使用 map 等。此外,在多线程环境中,可以使用 sync.Map 来代替普通的 map,保证线程安全。
来源:https://juejin.cn/post/7225959500839977017
猜你喜欢
- 如何显示一个等待或欢迎信息? <% Response.Buffer = True %
- 代码如下:<% myDSN="DSN=xur;uid=xur;pwd=xur"mySQL="s
- 译序:本文译自Smashingmagazine,但是原文讲述的内容有些浅,也不是很完整,前端观察在翻译的前提下,增加了更多的更系统的内容。如
- 阅读上一篇:成为一个顶级设计师的第一准则限制你的色彩成为一个顶级设计师的7个简单原则的第二部分限制使用你的色彩。好象上个准则是让你限制用你的
- 由于笔者最近在做一个跨数据库操作的测试,开始做IBatisNet (IBatis.DataMapper.1.6.2/IBatis.DataA
- 近段时间看了一些论坛上面关于分页的ASP程序依然有许多的关注者,但里面只有代码,没有详细的解释,对于初学者来说,这样总是得不到真正的掌握,此
- MySQL安装文件已被广泛应用但是也在不断的更新,这里介绍MySQL安装文件设置使用,帮助大家安装更新MySQL安装文件系统。Fedora5
- SQL Server 2000使得以XML导出数据变得更加简单,但在SQL Server 2000中导入XML数据并对其进行处理则有些麻烦。
- 有时在浏览网页时,常常因为网页中的图片文件过大而使下载时间较长,这样还没有下载完,就会有许多浏览者不耐烦地拂袖而去,从而损失了客户流。但要使
- 最近有Win10系统用户反映,由于自己的电脑安装有两个python软件,所以想要卸载掉其中一个,不过在卸载的时候却发现无法卸载,并且出现提示
- 从开始认识CSS(DW4)那时起,我就知道了CSS的强大,但从未用CSS排版过,因为我曾经尝试过学习,但感觉太难了而且用DW的表格,所见及所
- 现在我们常见到有些网站常有这样个功能:有个按钮(如工作职位或省份或地区的选择等等)你一点就跳出一个图层(对话框)让你选择之类的。关闭它就点右
- 两个多月来唯一一次有时间哄么么睡觉,我先给他讲了遍《从前有座山》,还是不睡。又给他讲了这个“保安的故事”:小A是名很敬业的保安,负责保护客户
- 1、ul标签在Mozilla中默认是有padding值的,而在IE中只有margin有值。 2、同一个的class选择符可以在一个
- 因为要用到过滤一组中重复的数据,使之变成没有重复的一组数据的功能,百度了一下,居然有朋友乱写,而且比较多,都没有认真测试过,只对字符可以,但
- 见下,把数字转成条形图、条形码的一个程序:<%Sub ShowChart(ByRef aValues,
- bookheader.asp Recommended Books for <%=session(&quo
- 本文中,abigale代表查询字符串,ada代表数据表名,alice代表字段名。技巧一:问题类型:ACCESS数据库字段中含有日文片假名或其
- 请问论坛的树状记录表是怎么展开的?如何做?论坛的这种展开技术一般采用两种方法实现,一种是采用递归的方法,优点是逻辑简单,编程简单,缺点是速度
- 有时候希望可以用js另开新窗口,但用window.open方法打开窗口总是被浏览器阻止,可以用下面的方法打开新窗口而不会遭到拦截1.新添加一