【C#】Dictionary底层实现

Dictionary的键值对映射是通过哈希算法的函数建立的。这里不对哈希算法做过多的解释,算法本身就是将不定长度的键(可以是整数、字符串或对象)转化成一个固定值。

字典的关键成员

csharp 复制代码
private struct Entry
{
    public int hashCode;
    public int next;
    public K key;//字典中键的类型
    public V value;//字典中值的类型
}

private int[] buckets;//哈希桶数组
private Entry[] entries;//数据实体数组

buckets作用:存储entries 数据实体数组的索引

存储原理(添加键值对)

1.通过哈希(Hash)算法获取键的哈希值,赋值到数据实体 中hashCode

2.将hashCode对buckets数组长度取余(%)获得buckets数组中的buckets下标

3.通过buckets下标获得buckets数组中的元素(即为entries下标)

a.元素有效(!= -1)代表buckets 的索引位置 存储了 entries数组中索引,将当前元素赋值给Entry数据实体的next,并将Entry数据实体的索引赋值给buckets数组中的元素

b.元素无效(== -1)代表buckets 的索引位置 没有 entries数组中索引,将Entry数据实体的索引赋值给buckets数组中的元素

存储例子

插入数据A

csharp 复制代码
Dictionary.Add("a", 123);
  1. 假设* 通过哈希(Hash)算法获取键("a")的值为10
  2. 将10对buckets数组长度取余(%)获得buckets数组中的索引2
  3. 通过 下标2 获得buckets数组中的buckets[2]
  4. buckets[2] == -1 代表buckets 的索引位置 没有 entries数组中索引,将Entry数据实体的索引赋值给buckets数组中的元素,即buckets[2] = 0

插入数据B

csharp 复制代码
Dictionary.Add("qwe", 456);
  1. 假设* 通过哈希(Hash)算法获取键("qwe")的值也为10
  2. 将10对buckets数组长度取余(%)获得buckets数组中的索引2
  3. 通过 下标2 获得buckets数组中的buckets[2]
  4. buckets[2] != -1 代表buckets 的索引位置 存储了 entries数组中索引,将当前元素赋值给Entry数据实体的next,即数据B的next = buckets[2] = 0,并将Entry数据实体的索引赋值给buckets数组中的元素,即buckets[2] = 1

读取原理(通过键获取值)

  1. 通过哈希(Hash)算法获取键的哈希值,赋值到数据实体 中hashCode
  2. 将hashCode对buckets数组长度取余(%)获得buckets数组中的buckets下标
  3. 通过buckets下标获得buckets数组中的元素(即为entries下标)
  4. 根据下标获取entries中的数据,判断Entry数据的key是否与想获取的相同
    a. 相同,返回Entry数据的value
    b. 不相同,根据Entry数据的next获取entries中的数据,直到key与想获取的相同,才返回value

读取例子

获取数据A

csharp 复制代码
Dictionary["a"]
  1. 假设* 通过哈希(Hash)算法获取键("a")的哈希值,赋值到数据实体为10
  2. 将10对buckets数组长度取余(%)获得buckets数组中的索引2
  3. 通过 索引2 获得buckets数组中的 buckets[2] = 1
  4. 根据 1 获取entries中的数据,判断 entries[1].key == "a"?
  5. (entries[1].key) = "qwe" != "a",根据Entry数据的next获取entries中的数据 entries[next] = entries[0],直到key与想获取的相同 entries[0].key == "a",才返回entries[0].value = 123

获取数据B

csharp 复制代码
Dictionary["qwe"]
  1. 假设* 通过哈希(Hash)算法获取键("a")的哈希值,赋值到数据实体为10
  2. 将10对buckets数组长度取余(%)获得buckets数组中的索引2
  3. 通过 索引2 获得buckets数组中的 buckets[2] = 1
  4. 根据 1 获取entries中的数据,判断 entries[1].key == "qwe"?
  5. 相同,返回entries[1].value = 456

因为作者精力有限,文章中难免出现一些错漏,敬请广大专家和网友批评、指正。

相关推荐
神仙别闹5 分钟前
基于 Java 语言双代号网络图自动绘制系统
java·开发语言
猫爪笔记12 分钟前
JAVA基础:单元测试;注解;枚举;网络编程 (学习笔记)
java·开发语言·单元测试
dangoxiba14 分钟前
[Unity Demo]从零开始制作空洞骑士Hollow Knight第十八集补充:制作空洞骑士独有的EventSystem和InputModule
游戏·unity·c#·游戏引擎·playmaker
API快乐传递者17 分钟前
用 Python 爬取淘宝商品价格信息时需要注意什么?
java·开发语言·爬虫·python·json
fengbizhe23 分钟前
qt获取本机IP和定位
开发语言·c++·qt·tcp/ip
yang_shengy27 分钟前
【JavaEE】认识进程
java·开发语言·java-ee·进程
无敌最俊朗@33 分钟前
unity3d————屏幕坐标,GUI坐标,世界坐标的基础注意点
开发语言·学习·unity·c#·游戏引擎
重生之我是数学王子38 分钟前
网络编程 UDP编程 Linux环境 C语言实现
linux·c语言·开发语言·网络·网络协议·udp
.net开发1 小时前
WPF使用Prism框架首页界面
前端·c#·.net·wpf
何曾参静谧1 小时前
「C/C++」C/C++标准库 之 #include<cstddef> 常用定义和宏
c语言·开发语言·c++