golang中map原理剖析
1. golang中的map有自己的一套实现原理,其核心是由hmap和bmap两个结构体实现的
2. 初始化map
package main func main() {
// 初始化一个可容纳10个元素的map
map1 := make(map[string]string, 10)
// 第一步:创建一个hmap结构体对象
// 第二步:生成一个哈希因子hash0并赋值到hmap对象中(用于后续为key创建哈希值)
// 第三步:根据hint=10,并根据算法规则来创建B,当前B应该为1
/*
hint B
0~8 0
9~13 1
14~26 2
...
*/
// 第四步:根据B去创建桶(bmap对象)并存在buckets数组中,当前bmap的数量应为2
// 当B<4时,根据B创建桶的个数的规则为:2的B次方(标准通)
// 当B>=4时,根据B创建桶的个数的规则为:2的B次方 + 2的(B-4)次方(标准通+溢出桶)
// 注意:每个bmap中可以存储8个键值对,当不够存时需要使用溢出桶,并将当前bmap中的overflow字段指向溢出桶的位置
}
3. map写入数据
map1["name"] = "武沛齐"
// 第一步:结合哈希因子和键name生成哈希值
// 第二步:取哈希值的后B位,并根据后B位的值来决定将此键值对存放到哪个桶中(bmap)。
/*
将哈希值和桶掩码(B个为1的二进制)进行&运算,最终得到哈希值的后B位的值,假设当B为1时,其结果为0
哈希值:0101001111001011010
桶掩码:0000000000000000001
结果: 0000000000000000000 = 0
通过示例你会发现,找桶的原则实际上是根据后B位的位运算计算出索引位置,然后再去buckets数组中根据索引找到目标桶(bmap)。
*/
// 第三步:在上一步确定桶之后,接下来就在桶中写入数据
/*
获取哈希值的tophash(即:哈希值的'高8位'),将tophash,key,value分别写入到桶中的三个数组中
如果桶已满,则通过overflow找到溢出桶,并在溢出桶中继续写入。‘
注意:以后在桶中查找数据时,会基于tophash来找,(tophash相同则再去比较key)
*/
// 第四步:hmap的个数count++(map中的元素个数+1)
4. map读取数据
value := map1["name"]
// 第一步:结合哈希因子和键name生成哈希值。
// 第二步:获取哈希值的后B位,并根据后B位的值来决定将此键存放到哪个桶中(bmap)。
// 第三步:确定桶之后,在根据key的哈希值计算出tophash(高8位),再根据tophash和key去桶中查找数据。
// 当前桶如果没有找到,则根据overflow在去溢出桶中找,均未找到则表示key不存在
5. map扩容
// 在向map中添加数据时,当达到某个条件,则会引发map扩容
// 扩容条件
// 1. map中数据总个数/桶个数>6.5时,引发翻倍扩容
// 2. 使用了太多的溢出桶时,(溢出桶使用的太多会导致map处理速度降低)
// B <= 15,已使用的溢出桶个数 >=2的B次方时,引发等量扩容
// B > 15,已使用的溢出桶个数 >=2的15次方时,引发等量扩容
6. map中数据迁移
golang中map原理剖析的更多相关文章
- golang 中 map 转 struct
golang 中 map 转 struct package main import ( "fmt" "github.com/goinggo/mapstructure&qu ...
- Golang中map的三种声明方式和简单实现增删改查
package main import ( "fmt" ) func main() { test3 := map[string]string{ "one": & ...
- golang中map并发读写问题及解决方法
一.map并发读写问题 如果map由多协程同时读和写就会出现 fatal error:concurrent map read and map write的错误 如下代码很容易就出现map并发读写问题 ...
- Go_14:GoLang中 json、map、struct 之间的相互转化
1. golang 中 json 转 struct <1. 使用 json.Unmarshal 时,结构体的每一项必须是导出项(import field).也就是说结构体的 key 对应的首字母 ...
- GoLang中 json、map、struct 之间的相互转化
1. golang 中 json 转 struct <1. 使用 json.Unmarshal 时,结构体的每一项必须是导出项(import field).也就是说结构体的 key 对应的首字母 ...
- 说说不知道的Golang中参数传递
本文由云+社区发表 导言 几乎每一个C++开发人员,都被面试过有关于函数参数是值传递还是引用传递的问题,其实不止于C++,任何一个语言中,我们都需要关心函数在参数传递时的行为.在golang中存在着m ...
- golang之map的使用声明
1.map的基本介绍 map是key-value数据结构,又称为字段或者关联数组.类似其它编程语言的集合,在编程中是经常使用到的 2.map的声明 1)基本语法 var map 变量名 map[key ...
- 0000 - Spring 中常用注解原理剖析
1.概述 Spring 框架核心组件之一是 IOC,IOC 则管理 Bean 的创建和 Bean 之间的依赖注入,对于 Bean 的创建可以通过在 XML 里面使用 <bean/> 标签来 ...
- Spring 中常用注解原理剖析
前言 Spring 框架核心组件之一是 IOC,IOC 则管理 Bean 的创建和 Bean 之间的依赖注入,对于 Bean 的创建可以通过在 XML 里面使用 <bean/> 标签来配置 ...
随机推荐
- IDEA推荐配置(自动导入包、提示不区分大小写)
设置快捷键方式为eclipse 设置代码提示不区分大小写 自动导入包 可以通过 Ctrl + 鼠标滚轮 来控制代码字体大小显示 显示行号和显示区分方法线 代码一行显示不下,软分行显示,点击鼠标右键 增 ...
- 【LeetCode】111. Minimum Depth of Binary Tree 解题报告(Python)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题方法 DFS BFS 日期 [LeetCode] 题目地址 ...
- 【LeetCode】119. 杨辉三角 II Pascal‘s Triangle II(Python & Java)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 目录 题目描述 题目大意 解题思路 方法一: 空间复杂度 O ( k ∗ ( k + 1 ...
- 【LeetCode】81. Search in Rotated Sorted Array II 解题报告(Python)
作者: 负雪明烛 id: fuxuemingzhu 个人博客: http://fuxuemingzhu.cn/ 题目地址:https://leetcode.com/problems/search-in ...
- 带你熟悉鸿蒙轻内核Kconfig使用指南
摘要:本文介绍了Kconfig的基础知识,和鸿蒙轻内核的图形化配置及进阶的使用方法. 本文分享自华为云社区<鸿蒙轻内核Kconfig使用笔记>,作者: zhushy. 1. Kconfig ...
- 亲测:三个值得练手的Java实战项目
测试奇谭,BUG不见. 大家好,我是谭叔. 一提到编码,很多小伙伴便感到头疼,特别是半路转行的小伙伴或者没有系统学习过计算机基础的小伙伴. 对于想学而不知道怎么学的小伙伴,我可以分享下我的策略: 刷一 ...
- MMD
目录 概 主要内容 定义 MMD for kernel function classes 一个无偏统计量 MMD test Borgwardt K., Gretton A., Rasch M., Kr ...
- rapidjson解析与构造实例
void rapidjson1(){ rapidjson::StringBuffer s; rapidjson::Writer<rapidjson::StringBuffer> write ...
- CS5213设计说明书|Capstone CS5213|CS5213设计参考电路
Capstone CS5213是一款HDMI到VGA转换器结合了HDMI输入接口和模拟RGB DAC输出且带支持片上音频数模转换器.CS5213芯片设计简单,整体芯片尺寸精悍,外围电路集成优化度较高, ...
- BL8810最新版规格书|BL8810方案|USB 2.0读卡器控制器
在数码产品越来越普及的今天,利用单反.手机.平板等产品随手拍下相片.储存一些非常重要的数据等已经成为很多人必做的事情,而除使用数据线进行数据导入导出的操作外,利用读卡器也是一个必然的选择,就以本人自己 ...