golang 基础 map及工厂函数
Map是一种数据结构,是一个集合,用于存储一系列无序的键值对。它基于键存储的,键就像一个索引一样,这也是Map强大的地方,可以快速快速检索数据,键指向与该键关联的值。
内部实现
Map是基于 散列表 来实现,就是我们常说的 Hash 表,所以我们每次迭代Map的时候,打印的Key和Value是无序的,每次迭代的都不一样,即使按照一定的顺序存在也不行。
这种方式的好处在于,存储的数据越多,索引分布越均匀,所以我们访问键值对的速度也就越快,当然存储的细节还有很多,大家可以参考Hash相关的知识,这里我们只要记住Map存储的是无序的键值对集合。
声明和初始化
make函数
dict := make(map[string]int)
示例中创建了一个键类型为string的,值类型为int的空map。
dict["张三"] = 43
存储了一个Key为张三的,Value为43的键值对数据。
Map字面量
此外还有一种使用map字面量的方式创建和初始化map,对于上面的例子,我们可以同等实现。
dict := map[string]int{"张三":43}
使用一个大括号进行初始化,键值对通过:分开,如果要同时初始化多个键值对,使用逗号分割。
创建nil map
[danger]nil的Map是未初始化的,可以只声明一个变量,不分配内存。
var dict map[string]int
这个 map是不能操作存储键值对的,必须要初始化后才可以,比如使用make函数, 为其开启一块可以存储数据的内存,也就是初始化。
var dict map[string]int
dict = make(map[string]int)
dict["张三"] = 43
fmt.Println(dict)
Map的键可以是任何值,键的类型可以是内置的类型,也可以是结构类型,但是不管怎么样,这个键可以使用
==运算符进行比较,所以像切片、函数以及含有切片的结构类型就不能用于Map的键了,因为他们具有引用的语义,不可比较。
使用Map
Map的使用很简单,和数组切片差不多,数组切片是使用索引,Map是通过键。
赋值或更新
dict := make(map[string]int)
dict["张三"] = 43
判断键是否存在
age := dict["张三"]
在Go Map中,如果我们获取一个不存在的键的值,也是可以的,返回的是值类型的零值,这样就会导致我们不知道是真的存在一个为零值的键值对呢,还是说这个键值对就不存在。对此,Map为我们提供了检测一个键值对是否存在的方法。
age, exists := dict["李四"]
和获取键的值没有太大区别,只是多了一个返回值。
- 第一个返回值是键的值;
- 第二个返回值标记这个键是否存在,这是一个boolean类型的变量
删除
删除一个Map中的键值对,可以使用Go内置的delete函数。
delete(dict,"张三")
delete函数接受两个参数,第一个是要操作的Map,第二个是要删除的Map的键。
delete函数删除不存在的键也是可以的,只是没有任何作用。
遍历和排序Map
使用for range风格的循环,和遍历切片一样。
dict := map[string]int{"张三": 43}
for key, value := range dict {
fmt.Println(key, value)
}
这里的 range 返回两个值
第一个是Map的键
第二个是Map的键对应的值。
这里再次强调,这种遍历是无序的,也就是键值对不会按既定的数据出现,如果想安顺序遍历,可以先对Map中的键排序,然后遍历排序好的键,把对应的值取出来,下面看个例子就明白了。
package main
import (
"sort"
"fmt"
)
func main() {
dict := map[string]int{"王五": 60, "张三": 43}
// 先对 names 的 slice 进行排序
var names []string
for name := range dict {
names = append(names, name)
}
// second method,当range 返回的第二个参数不需要使用时,可以舍弃,还是推荐第一种写法
/*
for name, _ := range dict {
names = append(names, name)
}
*/
sort.Strings(names)
// 按排序遍历
for _, key := range names {
fmt.Println(key, dict[key])
}
}
在函数间传递Map
map 是引用传递
func main() {
dict := map[string]int{"王五": 60, "张三": 43}
modify(dict)
fmt.Println(dict["张三"])
}
func modify(dict map[string]int) {
dict["张三"] = 10
}
// out
10
上面这个例子输出的结果是10,也就是说已经被函数给修改了,可以证明传递的并不是一个Map的副本。这个特性和切片是类似的,这样性能就会更高,因为复制整个Map的代价太大了。
和其他语言的差异
访问的 key 不存在时,仍会返回零值,不能通过 nil 来判断元素是否存在。
工厂函数
顾名思义,就好比一个工厂一样,可以批量制造某种类型的东西。其实说白了就是封装了个方法减少重复工作。
示例,通过map实现的计算平方,立方
mapx := map[int]func(n int) int{}
mapx[1] = func(n int) int { return n }
mapx[2] = func(n int) int { return n * n }
mapx[3] = func(n int) int { return n * n * n }
fmt.Println(mapx[1](2))
fmt.Println(mapx[2](2))
fmt.Println(mapx[3](2))
示例,go 实现 set 集合
// new empty set
set := make(map[string]bool)
// add
set["name"] = true
set["age"] = true
set["gender"] = true
// loop
for k := range set {
fmt.Println(k)
}
// delete
delete(set, "name")
// exist
_, exist := set["age"]
if exist == true {
fmt.Println("Exist")
} else {
fmt.Println("Not Exist")
}
// len
fmt.Println(len(set))
go set 参考
- https://studygolang.com/articles/20832 go 中一些set的实现及 bitset
- https://allenwu.itscoder.com/set-in-go 实现set的过程
golang 基础 map及工厂函数的更多相关文章
- golang基础--map字典
map 类似python语言中的dict(字典类型),以key-value形式存储数据 Key必须是支持==或!=比较运算的类型,不可以是函数,map或slice map查找比线性搜素快很多,但比使用 ...
- Golang基础之函数
golang基础之函数 1.为什么需要函数? 有些相同的代码可能出现多次,如果不进行封装,那么多次写入到程序中,会造成程序冗余,并且可读性降低 2.什么是函数 为完成某些特定功能的程序指令集合称为函数 ...
- GoLang基础数据类型--->字典(map)详解
GoLang基础数据类型--->字典(map)详解 作者:尹正杰 版权声明:原创作品,谢绝转载!否则将追究法律责任. 可能大家刚刚接触Golang的小伙伴都会跟我一样,这个map是干嘛的,是 ...
- jQuery基础的工厂函数以及定时器的经典案例
1. jQuery的基本信息: 1.1 定义: jQuery是JavaScript的程序库之一,它是JavaScript对象和实用函数的封装, 1.2 作用: 许多使用JavaScript能实现的交 ...
- jQuery基础,定时器,工厂函数
这个星期刚刚学的JQuery,下面我来说说我学的这几个例子 jQuery是JavaScript的一个程序库. Jquery的工厂函数$(): 在Jquery中 $符号等价于jquery,作用是将DOM ...
- Golang基础教程——map使用篇
本文始发于个人公众号:TechFlow,原创不易,求个关注 今天是golang专题的第7篇文章,我们来聊聊golang当中map的用法. map这个数据结构我们经常使用,存储的是key-value的键 ...
- python基础:内置函数zip,map,filter
一.zip zip,就是把俩list,合并到一起,如果想同时循环2个list的时候,可以用zip,会帮你轮流循环两个list 比如: l1=[1,2,3,4,5] l2=['a','b','c','d ...
- Golang 基础之基础语法梳理 (三)
大家好,今天将梳理出的 Go语言基础语法内容,分享给大家. 请多多指教,谢谢. 本次<Go语言基础语法内容>共分为三个章节,本文为第三章节 Golang 基础之基础语法梳理 (一) Gol ...
- Golang 基础之基础语法梳理 (一)
大家好,今天将梳理出的 Go语言基础语法内容,分享给大家. 请多多指教,谢谢. 本次<Go语言基础语法内容>共分为三个章节,本文为第一章节 Golang 基础之基础语法梳理 (一) Gol ...
随机推荐
- SQL Server中创建sde数据库
在ArcCatalog或者ArcMap中打开ArcToolBox工具箱. 在工具箱中,找到创建企业级地理数据库工具,依次为数据管理工具→地理数据库管理→创建企业级地理数据库,如图所示. 双击打开创建企 ...
- 图的广度优先遍历(bfs)
广度优先遍历: 1.将起点s 放入队列Q(访问) 2.只要Q不为空,就循环执行下列处理 (1)从Q取出顶点u 进行访问(访问结束) (2)将与u 相邻的未访问顶点v 放入Q, 同时将d[v]更新为d[ ...
- Unity 阴影淡入淡出效果中Shader常量 unity_ShadowFadeCenterAndType和_LightShadowData的问题
由于Universal Render Pipeline目前(2020年4月1日)把阴影淡入淡出这个功能竟然给取消了…我自己拿片元位置到相机位置的距离进行了一个淡化,但是阴影边缘老是被裁切…后来研究了一 ...
- [HDU2072]单词数<字符串>
链接:http://acm.hdu.edu.cn/showproblem.php?pid=2072 Problem Description lily的好朋友xiaoou333最近很空,他想了一件没有什 ...
- MATLAB——时间,日期及显示格式
一.日期和时间 1.生成指定格式日期和时间 标准日期格式 2.获取当前时间的数值 >> datestr(now,) ans = -- :: >> datestr(now,'yy ...
- 手工注入——access手工注入实战和分析
今天进行了access手工注入,下面是我的实战过程和总结. 实战环境使用的是墨者学院的在线靶场.下面咱们直接进入主题. 第一步,判断注入点 通过‘ 或者 and 1=1 和 and 1=2 是否报错, ...
- localStorage应用(写的时间缓存在本地浏览器)
最近用了下localStorage,于是想记录加深下映象: 有关更详细的介绍,可以去看https://www.cnblogs.com/st-leslie/p/5617130.html: 我这引用了这个 ...
- git设置
1:注册码云2:点击个人主页创建私有项目3:下载git4:点击码云 头像 选择下方的设置-->点击左侧的SSH公钥-->怎样生成公钥(linux操作) window系统可以右击选择 git ...
- Redis 笔记(二)—— STRING 常用命令
字符串中不仅仅可以存储字符串,它可以存储以下 3 中类型的值 : 字符串 整数 浮点数 Redis 可以对字符串进行截取等相关操作,对整数.浮点数进行增减操作. 自增自减命令 命令 用例和描述 INC ...
- New!一只菜鸟的学习之路....
今天拥有了自己的博客,希望在这里记录下自己成长的点点滴滴! 本博客主要记录: 1.在学习过程中遇到的问题及后续的解决办法: 2.技术上的困难,希望路过的大佬指点一二: 3.分享一些实用的技术材料: 4 ...