口述Map

1、Go语言Map的底层实现原理是怎样的?

Go语言中Map底层其实就是一个哈希表 ,也就是一个叫hmap的结构体。所以Map在运行时的表现为,一个指向hmap结构体的指针。

hmap这个结构体,内部有几个挺重要的字段,分别是Bflagscountbucketsoldbuckets...

其中bucketsoldbuckets是一个指针,指向对应的数组的地址。这些数组中放着的是,bmap的数据结构。

bmap也是一个结构体,其内可以存8个key-value的键值对 ,与8个tophash(高八位),还附带着一个指向下一个溢出桶overflow-bmap)的位置。

注:

存放 Key 值和 Value 值的时候是分开存放的。先存放 Key 值,然后紧接着把 8 个 Key 值存完之后,才会存放 Value 值。这样是为了内存对齐。方便快速存取,与避免空间浪费。

2、Go语言Map的遍历是有序的还是无序的?

Go语言中Map的遍历时无序的 ,他会从一个随机的桶 中的,随机出来的 槽位(slot)开始,向圆圈一样,遍历一整圈。因为每次都是随机的,所以无序。

3、Go语言Map的遍历为什么要设计成无序的?

这个其实是设计者,故意的设计。就是害怕某些人钻漏洞,把map当成排序来用。因为每次遍历桶之前,他都可能刚经历过等量或者增量扩容 ,导致原本的存储位置,被打乱,比如有的key-value还傻傻的在原地,而另一些key-value又被规则重新分配到了其他位置。

所以每次遍历Map,Go语言都会通过引入随机数,打乱开始遍历Map的起点。

4、Map如何实现顺序读取?

因为Map不支持顺序读取 ,所以可以先把Map内的键值对,重新取出来,另放到一个切片 中,然后sort排序一下。

5、Go语言的Map是否是并发安全的?

Map当然不是并发安全的

如果不同协程/线程一起读,若只是只读 还是没有问题的。

但,你还记得hmap内部有一个flags的标记位吗?他是int类型的。你对他新增的或者删除的时候,都会将标记位先置为4

这时你去读。Go语言就会直接给你来个Painc

6、Map的Key一定要是可比较的吗?为什么?

必须要可比较

因为要把key-value,存进map里面的时候,会先对key进行哈希计算 ,得出一个哈希值

然后拿着这个哈希值,取低B位 ,找到他所在的bmap桶。

然后再取这个哈希值的高8位 ,去找它他应该到的槽位(slot

而这个所谓的 B ,则是hmap哈希表结构体中的一个uint8类型的字段。代表整个哈希表一共有,2B次方的桶。比如B=3,则有23次方=8个桶。

注:旧版哈希冲突是这样解决的:

他先在主桶内的8个槽位 寻找空位,如果8个槽位满了,则会通过 overflow bucket ,去寻找接下来的溢出桶,如果没有就新建...往复循环。

7、Go语言Map的扩容时机是怎样的?

分为两种扩容,一种是增量扩容,一种是等量扩容
等量扩容

  • 官方上说,若负载因子大于6.5,则容量翻倍。(B+1了)

增量扩容

  • B小于15时,总溢出桶的个数大于2B次方,则等量扩容。
  • B大于等于15时,总溢出桶的个数一旦大于215次方,就等量扩容。

8、Go语言Map的扩容过程是怎样的?

因为一次性搬迁完的话,会导致当次的写入操作明显卡顿。所以Go采用了渐进式搬迁

就拿翻倍扩容 举例子,他第一次先不搬完。而是每次 插入 / 删除 / 修改 的时候,顺带着搬一点。保证了平滑性。

9、可以对Map的元素取地址吗?

go 复制代码
如:fmt.Println(&map[3])

当然不可以,因为某个key的具体位置,随时会变动,所以这在Go语言内是不允许的。

10、Map 中删除一个 key,它的内存会释放么?

删除map中一个key后,该entry会被清除;

若其关联对象不再可达,后续可被GC回收。

只是mapbucket空间通常不会因为一次delete立刻释放。他是要等GC的。

11、Map可以边遍历边删除吗

在同一个 goroutine 里是允许的

Go 语言规范明确允许在 range 遍历 map 时删除元素:如果某个尚未遍历到的 entry 被删掉了,它之后就不会被产出;

如果遍历过程中新增元素,则"可能被遍历到,也可能不会"。
但普通 map 不是并发安全的

如果一个 goroutine 在遍历或读取,另一个 goroutine 同时写入或删除,运行时经常会直接报错。

补充:

自定义快排

go 复制代码
package main

import (
	"fmt"
	"slices"
)

type Person struct {
	Name string
	Age int
}

func main() {
	ps := []Person{
		{"Tom", 23},
		{"Alice", 18},
		{"Bob", 20},
	}
	nums := []int{5, 2, 9, 1, 3}
	// 正序
	slices.Sort(nums)
	// 反序
	slices.Reverse(nums)
	// 自定义
	slices.SortFunc(ps, func(a, b Person) int {
		return a.Age - b.Age // 按年龄升序
	})

	fmt.Println(ps)
}
相关推荐
假如让我当三天老蒯8 小时前
前端跨域解决方案(学习用)
前端·javascript·面试
Colin草率地做慢慢地改8 小时前
关于QuickStore这个项目的重构(2)- 数据库建表文件
后端·面试·架构
JieE21218 小时前
LeetCode 56. 合并区间|超清晰 JS 图解思路,面试高频区间题
javascript·算法·面试
JustHappy1 天前
我汇总了身边朋友的经历才发现,其实第一份实习是最难找的......
前端·后端·面试
uhakadotcom1 天前
在python 的 工程化架构中 ,什么是 薄包装器层?
后端·面试·github
假如让我当三天老蒯1 天前
模块化:ES Module 与 CommonJS 的区别
前端·面试
沉默王二1 天前
面试官:RAG 不用向量数据库,用 MySQL 硬扛?我:100 万向量不是很轻松?
mysql·面试·ai编程
Darling噜啦啦2 天前
列表转树算法深度解析:从 Map 到 Reduce 的两种实现,面试高频考点
数据结构·算法·面试
swipe2 天前
正则表达式入门到进阶:从表单校验到手写模板引擎
前端·javascript·面试