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/> 标签来配置 ...
随机推荐
- docker部署验证码项目报错:at sun.awt.FontConfiguration.getVersion(FontConfiguration.java:1264)
如果docker部署启动报错 java.lang.NullPointerException: nullat sun.awt.FontConfiguration.getVersion(FontConfi ...
- 带你熟悉鸿蒙轻内核Kconfig使用指南
摘要:本文介绍了Kconfig的基础知识,和鸿蒙轻内核的图形化配置及进阶的使用方法. 本文分享自华为云社区<鸿蒙轻内核Kconfig使用笔记>,作者: zhushy. 1. Kconfig ...
- 慢 SQL 优化
# 导致SQL执行慢的原因 1. 硬件问题.如网络速度慢,内存不足,I/O吞吐量小,磁盘空间满了等. 2. 没有索引或者索引失效.(一般在互联网公司,DBA会在半夜把表锁了,重新建立一遍索引,因为当你 ...
- .NET 编码的基础知识
.NET 编码的一些基本概念和分析 简单的类型概念 Hex (16进制) byte 字节 范围是:0~255,二进制下的范围就是00000000~11111111,相当于1字节. byte[] 字节数 ...
- SpringCloud集成Security安全(Eureka注册中心)
1.说明 为了保护注册中心的服务安全, 避免恶意服务注册到Eureka, 需要对Eureka Server进行安全保护, 本文基于Spring Security方案, 为Eureka Server增加 ...
- MySQL客户端mysql常用命令
通过MySQL自带的mysql命令行工具, 执行MySQL的相关命令. 1.连接MySQL服务端 mysql -uUserName -pPassword -h HostName_IP -P 3306 ...
- 前端必备,5大mock省时提效小tips,用了提前下班一小时
一.一些为难前端的业务场景 在我的工作经历里,需要等待后端童鞋配合我的情形大概有以下几种: a.我们跟外部有项目合作,需要调用到第三方接口. 一般这种情况下,商务那边谈合同,走流程,等第三方审核, ...
- Linux密码文件介绍
1. 查看shadow文件内容```cat /etc/shadow```可以看到shadow文件内容,例如:```root:$1$Bg1H/4mz$X89TqH7tpi9dX1B9j5YsF.:148 ...
- Flask + flask_sqlalchemy + jq 完成书籍展示、新增、删除功能
后端代码 from flask import Flask, render_template, request, jsonify from flask_wtf.csrf import CSRFProte ...
- solr -window 安装与启动
1.下载 官网路径 https://solr.apache.org/downloads.html 为了稳定,我用 5.4.1 版本的 , 这是下载地址 https://archive.apac ...