分享好友 最新动态首页 最新动态分类 切换频道
Java入门——Map,Set和搜索树,哈希表
2024-12-26 14:46

Map和set是一种专门用来进行搜索的容器或者数据结构,其搜索的效率与其具体的实例化子类有关。

以前常见的 搜索方式有

1. 直接遍历,时间复杂度为O(N),元素如果比较多效率会非常慢

2. 二分查找,时间复杂度为O(log2N) ,但搜索前必须要求序列是有序的

上述排序比较适合静态类型的查找,即一般不会对区间进行插入和删除操作了,而现实中的查找比如

1. 根据姓名查询考试成绩

2. 通讯录,即根据姓名查询联系方式

3. 不重复集合,即需要先搜索关键字是否已经在集合中

可能在查找时进行一些插入和删除的操作,即动态查找,那上述两种方式就不太适合了,本节介绍的Map和Set是 一种适合动态查找的集合容器。

一般把搜索的数据称为关键字(Key,和关键字对应的称为值(Value,将其称之为Key-value的键值对,所以 模型会有两种

1. 纯 key 模型,比如: 有一个英文词典,快速查找一个单词是否在词典中 快速查找某个名字在不在通讯录中

2. Key-Value 模型,比如: 统计文件中每个单词出现的次数,统计结果是每个单词都有与其对应的次数:单词,单词出现的次数> 梁山好汉的江湖绰号:每个好汉都有自己的江湖绰号

而Map中存储的就是key-value的键值对,Set中只存储了Key。


Map是一个接口类,该类没有继承自Collection,该类中存储的是结构的键值对,并且K一定是唯一的,不能重复。

Map.Entry <K,V>是Map内部实现的用来存放键值对映射关系的内部类

该内部类中主要提供了<key,value> 的获取,value的设置以及Key的比较方式。

注意

1. Map是一个接口,不能直接实例化对象,如果要实例化对象只能实例化其实现类TreeMap或者HashMap

2. Map中存放键值对的Key是唯一的,value是可以重复的

3. 在Map中插入键值对时,key不能为空,否则就会抛NullPointerException异常,但是value可以为空

4. Map中的Key可以全部分离出来,存储到Set中来进行访问(因为Key不能重复)。

5. Map中的value可以全部分离出来,存储在Collection的任何一个子集合中(value可能有重复)。

6. Map中键值对的Key不能直接修改,value可以修改,如果要修改key,只能先将该key删除掉,然后再来进行 重新插入。

7. TreeMap和HashMap的区别

 

Set与Map主要的不同有两点:Set是继承自Collection的接口类,Set中只存储了Key。

注意

1. Set是继承自Collection的一个接口类

2. Set中只存储了key,并且要求key一定要唯一

3. Set的底层是使用Map来实现的,其使用key与Object的一个默认对象作为键值对插入到Map中的

4. Set最大的功能就是对集合中的元素进行去重

5. 实现Set接口的常用类有TreeSet和HashSet,还有一个LinkedHashSet,LinkedHashSet是在HashSet的基础 上维护了一个双向链表来记录元素的插入次序。

6. Set中的Key不能修改,如果要修改,先将原来的删除掉,然后再重新插入

7. Set中不能插入null的key。

8. TreeSet和HashSet的区别

 

叉搜索树又称二叉排序树,它或者是一棵空树,或者是具有以下性质的二叉树:

  • 若它的左子树不为空,则左子树上所有节点的值都小于根节点的值
  • 若它的右子树不为空,则右子树上所有节点的值都大于根节点的值
  • 它的左右子树也分别为二叉搜索树

1. 如果树为空树,即根 == null,直接插入

2. 如果树不是空树,按照查找逻辑确定插入位置,插入新结点

设待删除结点为 cur, 待删除结点的双亲结点为 parent

1. cur.left == null

1. cur 是 root,则 root = cur.right

2. cur 不是 root,cur 是 parent.left,则 parent.left = cur.right

3. cur 不是 root,cur 是 parent.right,则 parent.right = cur.right

2. cur.right == null

1. cur 是 root,则 root = cur.left

2. cur 不是 root,cur 是 parent.left,则 parent.left = cur.left

3. cur 不是 root,cur 是 parent.right,则 parent.right = cur.left

3. cur.left != null && cur.right != null

需要使用替换法进行删除,即在它的右子树中寻找中序下的第一个结点(关键码最小),用它的值填补到被 删除节点中,再来处理该结点的删除问题

 

顺序结构以及平衡树中,元素关键码与其存储位置之间没有对应的关系

因此在查找一个元素时,必须要经过关键码的多次比较。

顺序查找时间复杂度为O(N),平衡树中为树的高度,即O( log2N),搜索的效率取决于搜索过程中 元素的比较次数。

理想的搜索方法

可以不经过任何比较,一次直接从表中得到要搜索的元素。 如果构造一种存储结构,通过某种函 数(hashFunc)使元素的存储位置与它的关键码之间能够建立一一映射的关系,那么在查找时通过该函数可以很快 找到该元素。

当向该结构中

插入元素

根据待插入元素的关键码,以此函数计算出该元素的存储位置并按此位置进行存放

搜索元素

对元素的关键码进行同样的计算,把求得的函数值当做元素的存储位置,在结构中按此位置取元素比较,若 关键码相等,则搜索成功

该方式即为哈希(散列)方法,哈希方法中使用的转换函数称为哈希(散列)函数,构造出来的结构称为哈希表(Hash Table)(或者称散列表)

用该方法进行搜索不必进行多次关键码的比较,因此搜索的速度比较快

对于两个数据元素的关键字 Ki和Kj (i != j),有Ki !=Kj ,但有:Hash( Ki) == Hash(Kj ),即

不同关键字通过相同哈 希哈数计算出相同的哈希地址,该种现象称为哈希冲突或哈希碰撞。

把具有不同关键码而具有相同哈希地址的数据元素称为“同义词”。

首先,我们需要明确一点,由于我们哈希表底层数组的容量往往是小于实际要存储的关键字的数量的,这就导致一 个问题冲突的发生是必然的,但我们能做的应该是尽量的降低冲突率。

引起哈希冲突的一个原因可能是:哈希函数设计不够合理。

哈希函数设计原则

  1. 哈希函数的定义域必须包括需要存储的全部关键码,而如果散列表允许有m个地址时,其值域必须在0到m-1 之间
  2. 哈希函数计算出来的地址能均匀分布在整个空间中
  3. 哈希函数应该比较简单

闭散列也叫开放定址法,当发生哈希冲突时,如果哈希表未被装满,说明在哈希表中必然还有空位置,那么可以 把key存放到冲突位置中的“下一个” 空位置中去。那如何寻找下一个空位置呢

1. 线性探测

比如上面的场景,现在需要插入元素44,先通过哈希函数计算哈希地址,下标为4,因此44理论上应该插在该 位置,但是该位置已经放了值为4的元素,即发生哈希冲突。

线性探测:从发生冲突的位置开始,依次向后探测,直到寻找到下一个空位置为止。

插入

  • 通过哈希函数获取待插入元素在哈希表中的位置
  • 如果该位置中没有元素则直接插入新元素,如果该位置中有元素发生哈希冲突,使用线性探测找到 下一个空位置,插入新元素
  • 采用闭散列处理哈希冲突时,不能随便物理删除哈希表中已有的元素,若直接删除元素会影响其他 元素的搜索。比如删除元素4,如果直接删除掉,44查找起来可能会受影响。因此线性探测采用标 记的伪删除法来删除一个元素。

研究表明:当表的长度为质数且表装载因子a不超过0.5时,新的表项一定能够插入,而且任何一个位置都不 会被探查两次。因此只要表中有一半的空位置,就不会存在表满的问题。在搜索时可以不考虑表装满的情 况,但在插入时必须确保表的装载因子a不超过0.5,如果超出必须考虑增容。

因此闭散列最大的缺陷就是空间利用率比较低,这也是哈希的缺陷。

开散列法又叫链地址法(开链法),首先对关键码集合用散列函数计算散列地址,具有相同地址的关键码归于同一子 集合,每一个子集合称为一个桶,各个桶中的元素通过一个单链表链接起来,各链表的头结点存储在哈希表中。

从上图可以看出,开散列中每个桶中放的都是发生哈希冲突的元素。

开散列,可以认为是把一个在大集合中的搜索问题转化为在小集合中做搜索了。

1. HashMap 和 HashSet 即 java 中利用哈希表实现的 Map 和 Set

2. java 中使用的是哈希桶方式解决冲突的

3. java 会在冲突链表长度大于一定阈值后,将链表转变为搜索树(红黑树

4. java 中计算哈希值实际上是调用的类的 hashCode 方法,进行 key 的相等性比较是调用 key 的 equals 方 法。所以如果要用自定义类作为 HashMap 的 key 或者 HashSet 的值,必须覆写 hashCode 和 equals 方 法,而且要做到 equals 相等的对象,hashCode 一定是一致的。

最新文章
DS-K3B501S-L 海康威视智能安全摆闸 DS-K3B501S-M DS-K3B501S-R
DS-K3B501S-L 海康威视智能安全摆闸 采用12对红外检测传感器,具有高稳定性,支持人脸组件、各类IC/ID读卡器、二维码阅读器、指纹、身份的证阅读器等外设接入 遥控器DS-K7R01-868(国内标配) 采用直流无刷伺服电机 支持600万次无故障通行 设
2024年即热式热水器十大品牌
即热式热水器优缺点即热式热水器安全、小巧、美观,干净环保;即开即热,3-5秒出热水无须等候,热水使用时间不受限制,想用多久就用多久;用多少烧多少,省电省水,没有损耗;内置温控仪保证温度在30-50度之间,解决温度持续高温导致的结垢
IDC实测认证!深信服安全GPT现已落地超400家用户
  目前,深信服安全GPT已在超400家用户真实环境实践,已在超30家行业头部用户国家级攻防演练实战部署,某部委单位安全运营负责人反馈“安全GPT真正达到3-5年安全专家的水平”。  同时,国际权威机构IDC发布最新报告,再次印证深信服安全GPT
002027股票行情,002027股票历史行情
财经首页股票基金港股美股期货外汇贵金属债券期指权证我的自选行情中心。新浪财经意见反馈留言板电话:010-82628888-5174 欢迎批评指正。公告中加心享灵活配置混合型证券投资基金基金产品资料概要 基金资讯 06-29 09:06。F10档案:操盘必读
2024年氢气机十大品牌排行榜
中国报告大厅网的最新揭示了氢气机行业的品牌影响力。2024年,氢气机市场迎来了新的变化,各大品牌在产品质量、技术创新和市场占有率等方面展开了激烈的竞争。在2024年氢气机品牌排行榜中,各大品牌通过不断的努力和创新,提升了自身的品牌
AI设计——入门及快捷键知识点
对多个图形对象进行编组、要在画板中选中多个图形对象,然后执行对象/编组命令,得到编组对象。对象编组的快捷键是CTRL+G ,取消编组是CTRL+SHIFT+G。绘制线条图形线条分为直线段、弧线以及各种由线条组合的各种图形。根据要求选择不同的线
b站在线解析_用Python爬虫爬取B站用户在线人数,结果意料之外
好的,让我来回答你的问题。要使用 Python 爬虫爬取网易云音乐,可以使用 Requests 和 BeautifulSoup 库来发送 HTTP 请求并解析 HTML 页面。首先,你需要了解网易云音乐的 API 接口。你可以通过浏览器的开发者工具来查找这些接口。例如,在
AI绘画神器:一键生成超逼真美女写真教程
限时免费,点击体验最近超火的AI生图神器,坐拥3000美女的大男主就是你! https://ai.sohu.com/pc/generate/textToImg?_trans_=030001_yljdaimn 在这个科技迅猛发展的时代,AI技术已经渗透到了我们生活的每一个角落。想象一下,使用简单的A
4399游戏盒 3.7.0
4399游戏盒最新版是款由4399创造的小游戏大全。4399游戏盒ios版中不仅可以观看各类游戏直播,还可获取最新游戏资讯。4399游戏盒集游戏的推荐、资讯、社交、互动、管理于一体,具有免费高速下载游戏、好友互动、分享游戏状态、展示个人风采
2024年香港正版资料大全最新版,确保成语解释落实的问题_win305.210
随着香港特区政府对知识产权保护的重视程度不断加深,正版资料的推广和普及成为了香港社会的一个热点话题。2024年,香港正版资料大全最新版正式发布,旨在为公众提供一个全面、权威的正版资料查询平台,确保成语解释的落实。本文将详细介绍
相关文章
推荐文章
发表评论
0评