Golang中map数据类型的使用方法

目录
  • 前言
  • 案例
  • map
    • map定义
    • map声明
    • map的操作
  • 总结

前言

今天咱们来学习一下golang中的map数据类型,单纯的总结一下基本语法和使用场景,也不具体深入底层。map类型是什么呢?做过PHP的,对于数组这种数据类型是一点也不陌生了。PHP中的数组分为索引数组关联数组。例如下面的代码:

// 索引数组【数组的key是一个数字, 从0,1,2开始递增】
$array = [1, '张三', 12];
// 关联数组【数组的key是一个字符串,可以自定义key的名称】
$array = ['id' => 1, 'name' => '张三', 'age' => 12];

在golang中,map是一种特殊的数据结构,是一种key对应一个value类型的结构。这种结构可以被称为关联数组和字典。

在golang中也有切片和数组这样的数据类型,来存储一组数据。

  • 数组就好比PHP中的一维数组,并且长度是固定的,其中的值类型在定义数组的时候就确定好了。
  • 切片是一种特殊的数组类型。长度是固定的。
// 数组,类型是int,长度是4。
array := [4]int{1, 2, 3, 4}
// 切片,类型是int,长度不固定。
slice := []int{1, 2, 3, 4}

有数组和切片可以存储一组数据,那为什么还有map这样的类型结构呢?map类型具体是啥样的呢?

案例

假设我们现在有这样的一个需求,要用golang中的一种数据类型来存储多个用户的数据,这些数据分别用户的ID,name,age,sex…等等字段。我们改用什么数据类型呢?

在PHP中我们可以直接下面的方式定义,操作也是非常简单。

$userInfo = [
  ['id' => 1, 'name' => '张三', 'age' => 12, 'sex' => '男'],
  ['id' => 2, 'name' => '赵六', 'age' => 22, 'sex' => '男'],
  ['id' => 3, 'name' => '李四', 'age' => 34, 'sex' => '女'],
  ['id' => 4, 'name' => '王麻子', 'age' => 56, 'sex' => '男']
];

那如何在golang中实现呢,假设我们用数组和切片实现一下试试。

// 1. 用数组实现
$user1 := [4]string{"1", "张三", "12", "男"}
$user2 := [4]string{"2", "赵六", "12", "男"}
$user3 := [4]string{"3", "李四", "12", "女"}
$user4 := [4]string{"4", "王麻子", "12", "男"}
// 2. 用户切片实现
$user1 := []string{"1", "张三", "12", "男"}
$user2 := []string{"2", "赵六", "12", "男"}
$user3 := []string{"3", "李四", "12", "女"}
$user4 := []string{"4", "王麻子", "12", "男"}

通过上面的示例代码,我们是不是看得出来存在这样几个问题:

  • a. 可读性低。我们完全不知道1、12这样的值是用户的什么信息,男、张三我们还可以猜测一下是名字和性别。
  • b. 重复代码。一个用户一个变量,如果存在千万个用户,我们岂不是需要定义千万个变量。
  • c. 繁琐。相比PHP的实现,是不是非常繁琐。PHP中直接定义一个变量,通过多维数组的方式,就可以定义key和值。清晰并且简单。这也是为什么大家都说PHP中的数组非常强大和好用了。

通过切片和数组实现的方式,我们知道了弊端。那有不有一种数据类型能够像PHP这样简单就能实现呢?这样的场景就可以用map实现PHP这样的定义结构。接下来,我们就具体总结一下map相关的操作。

map

map定义

map 是一种特殊的数据结构:一种元素对(pair)的无序集合,pair 的一个元素是 key,对应的另一个元素是 value,所以这个结构也称为关联数组或字典。这是一种快速寻找值的理想结构:给定 key,对应的 value 可以迅速定位。
map 这种数据结构在其他编程语言中也称为字典(Python)、hash 和 HashTable 等。

map声明

map属于一种引用类型,在使用时我们需要make给其分配内存空间,未分配内存空间的map值是一个nil。

map声明时,需要指定key的类型和值的类型,并且复制时,必须按照定义时的类型进行复制。
map的值可以是任意类型,可以是切片可以是数组,可以是接口、结构体、指针、字符串等等数据类型。
var map1 map[key类型]值类型

// 声明方式一,完整模式
var map1 map[int]string
map1 = make(map[int]string[, n])

// 声明方式二,段语法风格
map1 := make(map[int]string[, n])

上面的n都是map的容量,也就是说map可以存储元素的数量。可以省略,map会动态扩容。
示例小案例,我们用map存储一个用户的信息。用户信息包含ID,name,age字段。

userInfo := make(map[string]string)
userInfo["id"] = "1"
userInfo["name"] = "张三"
userInfo["age"] = "12"
fmt.Println(userInfo)

// output
map[id:1 name:张三 age:12]

因为用户信息的字段和字段值有字符串和数字类型,定义好类型之后就只能传对应类型的值,因此我们给key和value的类型都定义为string类型。

map的操作

这里的操作,我们接着上面的小案例来使用。

  • 访问和复制。我们直接使用下标就可以了。
// 赋值
mapName[key] = "值"
userInfo["name"] = "王五"

// 访问
mapName[key]
name := userInfo["name"]
  • 删除元素。删除操作,需要使用到delete(),给该函数传递要删除的key。
delete(mapName, key)

delete(userInfo, "name")
  • 判断某一个值key是否存在。上面我们访问map中的key,直接使用下标就可以了。如果 map 中不存在 key1,val1 就是一个值类型的空值。会导致我们没法区分到底是 key不存在还是它对应的value就是空值。
value, boolean := mapName[key]
// 如果boolean是一个true则说明,对应的key存在,如果是false,则说明该key不存在。

if value, ok := userInfo["address"]; !ok {
  fmt.Println("address key not found!")
}
//因为address这个key不存在,因此会输出“address key not found!”。
  • 循环。循环map,我们一般是用到 for range来实现。
// 语法
for value, key := range mapName {
  fmt.Println(key, "=>", value)
}
// 示例
for value, key := range userInfo {
    fmt.Println(key, "=>", value)
}
// output
1 => id
张三 => name
12 => age

总结

其实对map基本的操作就是这么简单。对它的理解也是这么简单。在日常开发中,我们也经常使用该类型。

回到最上面多个用户的案例,这时候我们是不是就知道怎么使用map实现了。

  • 因为是多个用户,我们是不是需要定义多维的map结构。
  • map的一级key是int,表示当前的用户序号(从0,1,2,3…依次递增)。key对应的值,才是某一个用户的具体信息,我们同样的定义map类型来存储,key和value都是字符串,结构就像map声明中的小案例一样。
  • 因为我们不知道用户的具体个数,我们将一级的key定义为切片。因为切片的长度是不固定的。
userInfo := make([]map[string]string, 3)

for i := 0; i < 3; i++ {
    userInfo[i] = make(map[string]string, 3)
     userInfo[i]["id"] = "ID"
     userInfo[i]["name"] = "名称"
     userInfo[i]["age"] = "年龄"
}
fmt.Println(userInfo)

// output
[
  map[id:ID name:名称 age:年龄]
  map[id:ID name:名称 age:年龄]
  map[id:ID name:名称 age:年龄]
]

为什么要两次make,因为切片和map在golang中都是引用类型。第一次在make时,是针对切片初始化内存空间,第二次是针对切片的key对应的元素分配内存空间。大致的结构就像下图一样。

到此这篇关于Golang中map数据类型的使用方法的文章就介绍到这了,更多相关Golang map数据类型内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • golang并发工具MapReduce降低服务响应时间

    目录 前言 并发处理工具MapReduce MapReduce的用法演示 MapReduce使用注意事项 实现原理分析: 文末 前言 在微服务中开发中,api网关扮演对外提供restful api的角色,而api的数据往往会依赖其他服务,复杂的api更是会依赖多个甚至数十个服务.虽然单个被依赖服务的耗时一般都比较低,但如果多个服务串行依赖的话那么整个api的耗时将会大大增加. 那么通过什么手段来优化呢?我们首先想到的是通过并发来的方式来处理依赖,这样就能降低整个依赖的耗时,Go基础库中为我们提供

  • Golang编程并发工具库MapReduce使用实践

    目录 环境 项目需求 mapReduce使用说明 需求实现 业务逻辑 创建任务队列 运行结果 结论 引申阅读 环境 go version go1.16.4 windows/amd64 Intel(R) Core(TM) i7-7820HK CPU @ 2.90GHz 4核心8线程 项目需求 处理数个约5MB的小文件 从源目录读取文件并拷贝到目标目录 计算源文件MD5和目标文件MD5进行对比,如不相同则报错并终止程序执行 mapReduce使用说明 go get -u github.com/tal

  • golang 数组去重,利用map的实现

    目录 golang数组去重利用map golang删除排序数组中的重复项 golang数组去重利用map 可以利用go中,map数据类型的key唯一的属性,来对数组去重 将strSlice数组中重复的元素去掉,使其中的元素唯一 var strMap make(map[string]string) strSlice := []string {"slice","int","string","int","boolean&q

  • 深入了解Golang的map增量扩容

    目录 核心思想 扩容方式 源码分析 核心思想 以空间换时间,访问速度与填充因子有关 扩容hash表的时候每次都增大2倍,hash表大小始终为2的整数倍,有(hash mod 2^B) == (hash & (2^B-1)),方便于简化运算,避免取余操作 扩容前后的 hash mod 容量大小 是不等的,因此要重新计算每一项在hash表中的位置,扩容后需要将old pair重新hash到新的hash表上(就是一个evacuate的过程).这个过程不是一次性完成的,在每次insert.remove的

  • Golang中map数据类型的使用方法

    目录 前言 案例 map map定义 map声明 map的操作 总结 前言 今天咱们来学习一下golang中的map数据类型,单纯的总结一下基本语法和使用场景,也不具体深入底层.map类型是什么呢?做过PHP的,对于数组这种数据类型是一点也不陌生了.PHP中的数组分为索引数组和关联数组.例如下面的代码: // 索引数组[数组的key是一个数字, 从0,1,2开始递增] $array = [1, '张三', 12]; // 关联数组[数组的key是一个字符串,可以自定义key的名称] $array

  • Golang map实践及实现原理解析

    目录 Map实践以及实现原理 使用实例 内存模型 创建map hash函数 key定位和碰撞解决 扩容 元素访问 删除 迭代 Map实践以及实现原理 使用实例内存模型创建maphash函数key定位和碰撞解决扩容元素访问删除迭代核心点: 使用实例 测试的主要目的是对于map,当作为函数传参时候,函数内部的改变会不会透传到外部,以及函数传参内外是不是一个map,也就是传递的是实例还是指针.(golang里面的传参都是值传递). Test Case1:传参为map. func main(){ fmt

  • Golang中Map按照Value大小排序的方法实例

    目录 起因 探索 实现 第一步 第二步 第三步 总结 总结 Golang中的 map 默认是 无序的 . 起因 最近项目中有这样一个需求: 根据用户当前的坐标点,获取该用户附近的预设城市名称. 这里有一个注意点是,假设这些支持的城市名称是预设的,所以就不能直接通过地图类api根据坐标点获取所在城市名称了. 想到的解决思路是: 获取这几个预设城市的坐标点 App端获取用户当前坐标点 分别计算得到该用户坐标点距离各个预设城市的坐标点距离 然后计算得到其中距离最小的一项 这个坐标点对应的城市就是所求

  • JS自定义对象实现Java中Map对象功能的方法

    本文实例讲述了JS自定义对象实现Java中Map对象功能的方法.分享给大家供大家参考.具体分析如下: Java中有集合,Map等对象存储工具类,这些对象使用简易,但是在JavaScript中,你只能使用Array对象. 这里我创建一个自定义对象,这个对象内包含一个数组来存储数据,数据对象是一个Key,可以实际存储的内容!   这里Key,你要使用String类型,和Java一样,你可以进行一些增加,删除,修改,获得的操作. 使用很简单,我先把工具类给大家看下: 复制代码 代码如下: /**  *

  • golang 中获取字符串个数的方法

    在 golang 中不能直接用 len 函数来统计字符串长度,查看了下源码发现字符串是以 UTF-8 为格式存储的,说明 len 函数是取得包含 byte 的个数 // string is the set of all strings of 8-bit bytes, conventionally but not // necessarily representing UTF-8-encoded text. A string may be empty, but // not nil. Values

  • 浅析Java8 中 Map 接口的新方法

    我们提一个需求:给定一个 List<String> ,统计每个元素出现的所有位置. 比如,给定 list: ["a", "b", "b", "c", "c", "c", "d", "d", "d", "f", "f", "g"] ,那么应该返回: a : [

  • golang中sync.Mutex的实现方法

    目录 mutex 的实现思想 golang 中 mutex 的实现思想 mutex 的结构以及一些 const 常量值 Mutex 没有被锁住,第一个协程来拿锁 Mutex 仅被协程 A 锁住,没有其他协程抢锁,协程 A 释放锁 Mutex 已经被协程 A 锁住,协程 B 来拿锁 lockSlow() runtime_doSpin() runtime_canSpin() Mutex 被协程 A 锁住,协程 B 来抢锁但失败被放入等待队列,此时协程 A 释放锁 unlockSlow() Mutex

  • Golang中map的深入探究

    目录 简介 Map 的底层内存模型 Map 的存与取 底层代码 Map 的扩容 第一种情况 第二种情况 Map 的有序性 Map 的并发 总结 简介 本文主要通过探究在golang 中map的数据结构及源码实现来学习和了解map的特性,共包含map的模型探究.存取.扩容等内容.欢迎大家共同讨论. Map 的底层内存模型 在 golang 的源码中表示 map 的底层 struct 是 hmap,其是 hashmap 的缩写 type hmap struct { // map中存入元素的个数, g

  • 关于golang中map使用的几点注意事项总结(强烈推荐!)

    目录 前言 1 使用 map 记得初始化 2 map 的遍历是无序的 3 map 也可以是二维的 4 获取 map 的 key 最好使用这种方式 5 map 是并发不安全的 ,sync.Map 才是安全的 总结 前言 日常的开发工作中,map 这个数据结构相信大家并不陌生,在 golang 里面,当然也有 map 这种类型 关于 map 的使用,还是有蛮多注意事项的,如果不清楚,这些事项,关键时候可能会踩坑,我们一起来演练一下吧 1 使用 map 记得初始化 写一个 demo 定义一个 map[

  • Go中map数据类型3点小知识

    1.map数据类型初始化 两种方式:map[string]string{}或make(map[string]string) 2.未初始化的map是nil,它与一个空map基本等价,只是nil的map不允许往里面添加值.(A nil map is equivalent to an empty map except that no elements may be added) 因此,map是nil时,取值是不会报错的(取不到而已),但增加值会报错. 其实,还有一个区别,delete一个nil map

  • 初步解读Golang中的接口相关编写方法

    概述 如果说goroutine和channel是Go并发的两大基石,那么接口是Go语言编程中数据类型的关键.在Go语言的实际编程中,几乎所有的数据结构都围绕接口展开,接口是Go语言中所有数据结构的核心. Go语言中的接口是一些方法的集合(method set),它指定了对象的行为:如果它(任何数据类型)可以做这些事情,那么它就可以在这里使用. 接口的定义和使用 比如 复制代码 代码如下: type I interface{     Get() int     Put(int)   } 这段话就定

随机推荐