【数据结构与算法——TypeScript】
哈希表(HashTable)
哈希表介绍和特性
-
哈希表是一种非常重要的数据结构,但是很多学习编程的人一直搞不懂哈希表到底是如何实现的。
- 在这一章节中,我门就一点点来实现一个自己的哈希表。
- 通过实现来理解哈希表背后的原理和它的优势。
-
几乎所有的编程语言都有直接或者间接的应用这种数据结构。
-
哈希表通常是基于数组进行实现的,但是相对于数组,它也很多的优势:
- 口 它可以提供非常快速的插入-甽除-查找操作:
- 口 无论多少数据,插入和删除值都接近常量的时间:即O(1)的时间与杂度。实际上,只需要几个机器指令即可完成;
- 口 哈希表的速度比树还要快,基本可以瞬间查找到想要的元素:
- 口 哈希表相对于树来说编码要容易很多;
-
哈希表相对于数组的一些不足:
- 口 哈希表中的数据是没有顺序的,所以不能以一种固定的方式(出如从小到大)来遍力其中的元素(没有特妹处理情况下)
- 口 通常情况下,哈希表中的key是不允许重复的,不能放置相同的key,用于保存不同的元素。
哈希表到底是什么呢?
- 口 我们只是说了一下它的优势,似乎还是没有说它到底长什么样子?
- 口 这也是哈希表不好理解的地方,不像数组和链表,甚至是树一样直接画出你就知道它的结构,甚至是原理了。
- 口 ❤️ 它的结构就是数组,但是它神奇的地方在于对数组下标值的一种变换, 这种变换我们可以使用哈希函数,通过哈希函数可以获取到 HashCode.
- 口 不着急,我们慢慢来认识它到底是什么。
我们通过二个案例,案例需要你挑选某种数据结构,而你会发现最好的 选择就是哈希表
- 口 案例一:公司使用一种数据结构来保存所有员工;
- 口 案例二:使用一种数据结构存储单词信息,比如有50000个单词。找到单词后每个单词有自己的翻译&读音&应用等等:
案例一:公司员工存储
-
案例介绍:
- 口 假如一家公司有1000个员工,现在我们需要将这些员工的信息使用某种数据结构来保存起来
- 口 你会采用什么数据结构呢?
-
✅ 方案一:数组
- 口 一种方案是按照顺序将所有的员工依次存入一个长度为1000的数组中。
- 口 每个员工的信息都保存在数组的某个位置上。
- 口 但是我们要查看某个具体员工的信息怎么办呢?一个个找吗?不太好找。
- 口 数组最大的优势是什么?通过下标值去获取信息。
- 口 所以为了可以通过数组快速定位到某个员工,最好给员工信息中添加一个员工編号(工号), 而编号对应的就是员工的下标值。
- 口 当查找某个员工的信息时,通过员工编号可以快速定位到员工的信息位置,
-
✅ 方案二:链表
- 口 链表对应插入和删除数据有一定的优势。
- 口 但是对于获取员工的信息,每次都必须从头遍历到尾,这种方式显然不是特别适合我们这里。
-
✅ 最终方案:
- 口 这样看最终方案似乎就是数组了, 但是数组还是有缺点,什么缺点呢?
- 口 但是如果我们只知道员工的姓名, 比如why,但是不知道why的员工編号,你怎么办呢?
- 只能线性查找?效率非常的低
- 能不能有一种办法,让why的名字和它的员工编号产生直接的关系呢?
- 也就是通过why这个名字,我们就能获取到它的素引值,而再通过索引值我就能获取到why的信息呢?
- 口 这样的方案已经存在了,就是使用哈希函数,让某个key的信息和索引值对应起来。
案例二:50000个单词的存储
- 案例介绍:
- 口 使用一种数据结构存储单词信息,比如有50000个单词.
- 口 找到单词后每个单词有自己的翻译&读音&应用等等
- ✅ 方案一:数组?
- 口 这个案例更加明显能感受到数组的缺陷
- 口 拿到一个单词 lridescent,我想知道这个单词的翻译/读音/应用。
- 口 怎么可以从数组中查到这个单词的位置呢?
- 口 线性查找?50000次比较?
- 口 如果你使用数组来实现这个功能,效率会非常非常低,而且你一定没有学习过数据结构。
- ✅ 方案二:链表?
- ✅ 方案三:有没有一种方案,可以将单词转成数组的下标值呢
- 口 如果单词转成数组的下标,那么以后我们要查找某个单词的信息,直接按照下标值一步即可访问到想要的元素
❤️ 字母转数字的方案
似乎所有的案例都指向了一目标:将字符串转成下标值
但是,怎样才能将一个字符串转成数组的下标值呢?
口 单词/字符串转下标值,其实就是字母/文字转数字
口 怎么转?
-
现在我们需要设计一种方案,可以将单词转成适当的下标值:
- 口 其实计算机中有很多的编码方案就是用数字代皆单词的字符。就是字符编码。(常见的字符编码?)
- 口 比如ASCII编码:a是97,b是98,依次类推122代表z
- 口 我们也可以设计一个自己的编码系统,比如a是1,b是2,c是3,依 次类推,z是26。
- 口 当然我们可以加上空格用0代替,就是27个字符(不考虑大写问题)
- 口 但是,有了编码系统后,一个单词如何转成数字呢?
-
✅ 方案一:数字相加
- 口 一种转换单词的简单方案就是把单词每个字符的编码求和。
- 口 例如单词cats转成数字:3+1+20+19=43.
- 那么43就作为cats单词的下标存在数组中。
- 【问题】:按照这种方案有一个很明显的问题就是很多单词最终的下标可能都是43。
- 口 比如was/tin/give/tend/moan/tick等等。
- 口 我们知道数组中一个下标值位置只能存储一个数据如果存入后来的数据,必然会造成数据的獲盖。
- 一个下标存储这么多单词显然是不合理的。
- 口 虽然后面的方案也会出现,但是要尽虽避免。
-
✅ 方案二:幂的连乘
- 口 现在,我们想通过一种算法,让cats转成数字后不那么普通。
- 口 数字相加的方案就有些过于普通了。
- 口 有一种方案就是使用幂的连乘,什么是幂的连乘呢?
- 其实我们平时使用的大于10的数字,可以用一种系的连乘来表示它的唯一性:
比如:7654 = 7 *103 + 6 * 102 + 5 * 10 + 4
-
口 我们的单词也可以使用这种方案来表示:
- 比如 cats= 3273+1272+20*27+17= 60337
-
这样得到的数字可以基本保证它的唯一性,不会和别的单词重复。
-
【问题】:如果一个单词是 zzzzzzzzzz(一般英文单词不会超过10个宇符)。那么得到的数字超过7000000000000。
- 口 数组可以表示这么大的下标值吗?
- 口 而且就算能创建这么大的数组,事实上有很多是无效的单词。
- 口 创建这么大的数组是没有意义的
-
两种方案总结:
- 第一种方案(将数字相加求和)产生的数组下标太少
- 第二种方案(与27的幂相乘求和)产生的数组下标太多
数据的哈希化过程
下标的压缩算法
哈希表的一些概念
地址冲突解决方案
什么是冲突?
- ◼ 尽管50000个单词,我们使用了100000个位置来存储,并且通过一种相对比较好的哈希函数来完成。但是依然有可能会发生冲突。
- 比如 melioration 这个单词,通过哈希函数得到它数组的下标值后,发现那个位置上已经存在一个单词demystify
- 因为它经过哈希化后和melioration得到的下标实现相同的。
- ◼ 这种情况我们成为冲突。
- ◼ 虽然我们不希望这种情况发生,当然更希望每个下标对应一个数据项,但是通常这是不可能的。
- ◼ 冲突不可避免,我们只能解决冲突。
- : 就像之前0~199的数字选取5个放在长度为10的单元格中
- 如果我们随机选出来的是33,82,11,45,90,那么最终它们的位置会是3-2-1-5-0,没有发生冲突。
- 但是如果其中有一个33,还有一个73呢?还是发生了冲突。
- [外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传(img-wMwdr2p6-1691473209048)(/Volumes/web/数据结构与算法(ts)/画图/哈希表/取余操作.png “发生冲突”)]
- ◼ 我们需要针对 这种冲突 提出一些解决方案
- 虽然冲突的可能性比较小,你依然需要考虑到这种情况
- 以便发生的时候进行对应的处理代码。
- ◼ 如何解决这种冲突呢?常见的情况有两种方案。
链地址法
链地址法解析
- 图片解析:
- 从图片中我们可以看出,链地址法解决冲突的办法是每个数组单元中存储的不再是单个数据,而是一个链条。
- 这个链条使用什么数据结构呢?常见的是数组或者链表。
- 比如是链表,也就是每个数组单元中存储着一个链表。一旦发现重复,将重复的元素插入到链表的首端或者末端即可。
- 当查询时,先根据哈希化后的下标值找到对应的位置,再取出链表,依次查询找寻找的数据。
- 数组还是链表呢?
- 数组或者链表在这里其实都可以,效率上也差不多。
- 因为根据哈希化的index找出这个数组或者链表时,通常就会使用线性查找,这个时候数组和链表的效率是差不多的。
- 当然在某些实现中,会将新插入的数据放在数组或者链表的最前面,因为觉得新插入的数据用于取出的可能性更大。
- 这种情况最好采用链表,因为数组在首位插入数据是需要所有其他项后移的,链表就没有这样的问题。
- 当然,我觉得出于这个也看业务需求,不见得新的数据就访问次数会更多:比如我们微信新添加的好友,可能是刚认识的,联系的频率不见得比我们的老朋友更多,甚至新加的只是聊一两句。
- 所以,这里个人觉得选择数组或者链表都是可以的。
开放地址法
开放地址法解析
- 问题: 新插入32放在什么位置呢?
- 开放地址解决方法:
- 新插入的 32 本来应该插入到82的位置,但是该位置已经包含数据
- 我们发现 3 和 5 还有 9 的位置是没有任何内容的
- 这个时候就可以寻找到对应的空白位置来放这个数据
- 但是到底使用哪一个位置呢? 这就又需要分情况了
- 图片解析
- 从图片的文字中我们可以了解到
- 开放地址法其实就是要寻找空白的位置来放置冲突的数据项。
- 但是探索这个位置的方式不同,有三种方法:
线性探测
-
◼ 线性探测非常好理解:线性的查找空白的单元。
-
◼ 插入的32:
- 经过哈希化得到的index=2,但是在插入的时候,发现该位置已经有了82。怎么办呢?
- 线性探测就是从index位置+1开始一点点查找合适的位置来放置32,什么是合适的位置呢?
- 空的位置就是合适的位置,在我们上面的例子中就是index=3的位置,这个时候32就会放在该位置。
-
◼ 查询32呢?
- 查询32和插入32比较相似。
- 首先经过哈希化得到index=2,比如2的位置结果和查询的数值是否相同,相同那么就直接返回。
- 不相同呢?线性查找,从index位置+1开始查找和32一样的。
- 这里有一个特别需要注意的地方:如果32的位置我们之前没有插入,是否将整个哈希表查询一遍来确定32存不存在吗?
- 当然不是,查询过程有一个约定,就是查询到空位置,就停止。
- 因为查询到这里有空位置,32之前不可能跳过空位置去其他的位置。
-
线性探测的问题
- ◼ 删除32呢?
- 删除操作和插入查询比较类似,但是也有一个特别注意点。
- 注意:删除操作一个数据项时,不可以将这个位置下标的内容设置为null,为什么呢?
- 因为将它设置为null可能会影响我们之后查询其他操作,所以通常删除一个位置的数据项时,我们可以将它进行特殊处理(比如设置为-1)。
- 当我们之后看到-1位置的数据项时,就知道查询时要继续查询,但是插入时这个位置可以放置数据。
- 线性探测的问题
- 线性探测有一个比较严重的问题,就是聚集。什么是聚集呢?
- 比如我在没有任何数据的时候,插入的是22-23-24-25-26,那么意味着下标值:2-3-4-5-6的位置都有元素。
- 这种一连串填充单元就叫做聚集。
- 聚集会影响哈希表的性能,无论是插入/查询/删除都会影响。
- 比如我们插入一个32,会发现连续的单元都不允许我们放置数据,并且在这个过程中我们需要探索多次。
- 二次探测可以解决一部分这个问题,我们一起来看一看。
二次探测
- 我们刚才谈到,线性探测存在的问题:
- 如果之前的数据是连续插入的,那么新插入的一个数据可能需要探测很长的距离。
- ✓ 二次探测在线性探测的基础上进行了优化:
- 二次探测主要优化的是探测时的步长,什么意思呢?
- 线性探测,我们可以看成是步长为1的探测,比如从下标值x开始,那么线性测试就是x+1,x+2,x+3依次探测。
- 二次探测,对步长做了优化,比如从下标值x开始,x+1²,x+2²,x+3²。
- 这样就可以一次性探测比较长的距离,比避免那些聚集带来的影响。
- 二次探测的问题:
- 但是二次探测依然存在问题,比如我们连续插入的是32-112-82-2-192,那么它们依次累加的时候步长的相同的。
- 也就是这种情况下会造成步长不一样的一种聚集。还是会影响效率。(当然这种可能性相对于连续的数字会小一些)
- 怎么根本解决这个问题呢?让每个人的步长不一样,一起来看看再哈希法吧。
再哈希法
- ▪️ 为了消除线性探测和二次探测中无论步长+1还是步长+平法中存在的问题, 还有一种最常用的解决方案: 再哈希法.
- 再哈希法:
- 二次探测的算法产生的探测序列步长是固定的: 1, 4, 9, 16, 依次类推.
- 现在需要一种方法: 产生一种依赖关键字的探测序列, 而不是每个关键字都一样.
- 那么, 不同的关键字即使映射到相同的数组下标, 也可以使用不同的探测序列.
- 再哈希法的做法就是: 把关键字用另外一个哈希函数, 再做一次哈希化, 用这次哈希化的结果作为步长.
- 对于指定的关键字,步长在整个探测中是不变的, 不过不同的关键字使用不同的步长.
- 第二次哈希化需要具备如下特点:
- 和第一个哈希函数不同. (不要再使用上一次的哈希函数了, 不然结果还是原来的位置)
- 不能输出为0(否则, 将没有步长. 每次探测都是原地踏步, 算法就进入了死循环)
- 其实, 我们不用费脑细胞来设计了, 计算机专家已经设计出一种工作很好的哈希函数:
stepSize = constant - (key % constant)
- 其中constant是质数, 且小于数组的容量.
- 例如: stepSize = 5 - (key % 5), 满足需求, 并且结果不可能为0.
哈希化的效率
- 哈希表中执行插入和搜索操作效率是非常高的
- 如果没有产生冲突,那么效率就会更高。
- 如果发生冲突,存取时间就依赖后来的探测长度。
- 平均探测长度以及平均存取时间,取决于填装因子,随着填装因子变大,探测长度也越来越长。
- 随着填装因子变大,效率下降的情况,在不同开放地址法方案中比链地址法更严重,所以我们来对比一下他们的效率,再决定我们选取的方案。
- ◼ 在分析效率之前,我们先了解一个概念:装填因子
- 装填因子表示当前哈希表中已经包含的数据项和整个哈希表长度的比值。
- 装填因子 = 总数据项 / 哈希表长度。
- 开放地址法的装填因子最大是多少呢?1,因为它必须寻找到空白的单元才能将元素放入。
- 链地址法的装填因子呢?可以大于1,因为拉链法可以无限的延伸下去,只要你愿意。(当然后面效率就变低了)
线性探测效率
二次探测和再哈希化性能
链地址法性能
-
◼ 链地址法的效率分析有些不同,一般来说比开放地址法简单。我们来分析一下这个公式应该是怎么样的。
- 假如哈希表包含arraySize个数据项,每个数据项有一个链表,在表中一共包含N个数据项。
- 那么,平均起来每个链表有多少个数据项呢?非常简单,N / arraySize。
- 有没有发现这个公式有点眼熟?其实就是装填因子。
-
◼ OK,那么我们现在就可以求出查找成功和不成功的次数了
- 成功可能只需要查找链表的一半即可:1 + loadFactor/2
- 不成功呢?可能需要将整个链表查询完才知道不成功:1 + loadFactor。
-
◼ 经过上面的比较我们可以发现,链地址法相对来说效率是好于开放地址法的。
-
◼ 所以在真实开发中,使用链地址法的情况较多
- 因为它不会因为添加了某元素后性能急剧下降。
- 比如在Java的HashMap中使用的就是链地址法。
哈希函数代码实现
哈希函数
- 好的哈希函数应该尽可能让计算的过程变得简单,提高计算的效率。
- 哈希表的主要优点是它的速度,所以在速度上不能满足,那么就达不到设计的目的了。
- 提高速度的一个办法就是让哈希函数中尽量少的有乘法和除法。因为它们的性能是比较低的。
- 设计好的哈希函数应该具备哪些优点呢?
- ✅ 快速的计算
- ✓ 哈希表的优势就在于效率,所以快速获取到对应的hashCode非常重要。
- ✓ 我们需要通过快速的计算来获取到元素对应的hashCode
- ✅ 均匀的分布
- ✓ 哈希表中,无论是链地址法还是开放地址法,当多个元素映射到同一个位置的时候,都会影响效率。
- ✓ 所以,优秀的哈希函数应该尽可能将元素映射到不同的位置,让元素在哈希表中均匀的分布。
快速计算 - 霍纳法则
- ◼ 在前面,我们计算哈希值的时候使用的方式: ➡️ cats = 327³+127²+20*27+17= 60337
- ◼ 这种方式是直观的计算结果,那么这种计算方式会进行几次乘法几次加法呢?
- 当然,我们可能不止4项,可能有更多项
- 我们抽象一下,这个表达式其实是一个多项式: a(n)xn+a(n-1)x(n-1)+…+a(1)x+a(0)
- ◼ 现在问题就变成了多项式有多少次乘法和加法:
- 乘法次数:n+(n-1)+…+1=n(n+1)/2
- 加法次数:n次
- O(N²)
❤️
均匀分布
- ◼ 均匀的分布
- 在设计哈希表时,我们已经有办法处理映射到相同下标值的情况:链地址法或者开放地址法。
- 但是无论哪种方案,为了提供效率,最好的情况还是让数据在哈希表中均匀分布。
- 因此,我们需要在使用常量的地方,尽量使用质数。
- 哪些地方我们会使用到常量呢?
- ◼ 质数的使用:
- 哈希表的长度。
- N次幂的底数(我们之前使用的是27)
- ◼ 为什么他们使用质数,会让哈希表分布更加均匀呢?
- 质数和其他数相乘的结果相比于其他数字更容易产生唯一性的结果,减少哈希冲突。
- Java中的N次幂的底数选择的是31,是经过长期观察分布结果得出的;
N次幂的底数
- ◼ 这里采用质数的原因是为了产生的数据不按照某种规律递增。
- 比如我们这里有一组数据是按照4进行递增的:0 4 8 12 16,将其映射到长度为8的哈希表中。
- 它们的位置是多少呢?0 - 4 - 0 - 4,依次类推。
- 如果我们哈希表本身不是质数,而我们递增的数量可以使用质数,比如5,那么 0 5 10 15 20
- 它们的位置是多少呢?0 - 5 - 2 - 7 - 4,依次类推。也可以尽量让数据均匀的分布。
- 我们之前使用的是27,这次可以使用一个接近的数,比如31/37/41等等。一个比较常用的数是31或37。
- ◼ 总之,质数是一个非常神奇的数字。
- ◼ 这里建议两处都使用质数:
哈希函数的实现
function hashFunc(key: string, max: number): number {
let hascode = 0;
const length = key.length;
for (let i = 0; i < length; i++) {
hascode = 31 * hascode + key.charCodeAt(i);
}
const index = hascode % max;
return index;
}
console.log(hashFunc('abc', 7));
console.log(hashFunc('nba', 7));
console.log(hashFunc('cbd', 7));
console.log(hashFunc('mba', 7));
console.log(hashFunc('dfc', 7));
console.log(hashFunc('rft', 7));
哈希表创建和操作
创建哈希表
插入/修改数据put
-
哈希表的插入和修改操作是同一个函数:
- 因为,当使用者传入一个时
- 如果原来不存该key,那么就是插入操作。
- 如果已经存在该key,那么就是修改操作。
-
代码解析:
-
步骤1:根据传入的key获取对应的hashCode,也就是数组的index
-
步骤2:从哈希表的index位置中取出桶(另外一个数组)
-
步骤3:查看上一步的bucket是否为null
- 为null,表示之前在该位置没有放置过任何的内容,那么就新建一个数组[]
-
步骤4:查看是否之前已经放置过key对应的value
- 如果放置过,那么就是依次替换操作,而不是插入新的数据。
- 我们使用一个变量override来记录是否是修改操作
-
步骤5:如果不是修改操作,那么插入新的数据。
- 在bucket中push新的[key,value]即可。
- 注意:这里需要将count+1,因为数据增加了一项。
-
put的流程图
获取数据 get
- 获取数据:
-
根据key获取对应的value
-
代码解析:
- 步骤1:根据key获取hashCode(也就是index)
- 步骤2:根据index取出bucket。
- 步骤3:因为如果bucket都是null,那么说明这个位置之前并没有插入过数据。
- 步骤4:有了bucket,就遍历,并且如果找到,就将对应的value返回即可。
- 步骤5:没有找到,返回null
get(key: string): T | undefined {
const index = this.hashFunc(key, this.length);
const bucket = this.storage[index];
if (!bucket) return undefined;
for (let i = 0; i < bucket.length; i++) {
const tuple = bucket[i];
const tupleKey = tuple[0];
const tupleVal = tuple[1];
if (key === tupleKey) {
return tupleVal;
}
}
return undefined;
}
删除数据
-
删除数据:
- 我们根据对应的key,删除对应的key/value
-
◼ 代码解析:
delete(key: string): T | undefined {
const index = this.hashFunc(key, this.length);
const bucket = this.storage[index];
if (!bucket) return undefined;
for (let i = 0; i < bucket.length; i++) {
const tuple = bucket[i];
const tupleKey = tuple[0];
const tupleVal = tuple[1];
if (key === tupleKey) {
bucket.splice(i, 1);
this.count--;
return tupleVal;
}
}
return undefined;
}
哈希表的自动扩容
哈希表扩容的思想
- ❓为什么需要扩容?
- 目前,我们是将所有的数据项放在长度为7的数组中的。
- 因为我们使用的是链地址法,loadFactor可以大于1,所以这个哈希表可以无限制的插入新数据。
- 但是,随着数据量的增多,每一个index对应的****bucket会越来越长,也就造成效率的降低。
- 所以,在合适的情况对数组进行扩容,比如扩容两倍。
- ❓❓如何进行扩容?
- 扩容可以简单的将容量增大两倍(不是质数吗?质数的问题后面再讨论)
- 但是这种情况下,所有的数据项一定要同时进行修改(重新调用哈希函数,来获取到不同的位置)
- 比如hashCode=12的数据项,在length=8的时候,index=4。在长度为16的时候呢?index=12。
- 这是一个耗时的过程,但是如果数组需要扩容,那么这个过程是必要的。
-❓❓ 什么情况下扩容呢?
- 比较常见的情况是loadFactor>0.75的时候进行扩容。
- 比如Java的哈希表就是在装填因子大于0.75的时候,对哈希表进行扩容。
扩容函数
- 代码解析:
- 步骤1:先将之前数组保存起来,因为我们待会儿会将storeage = []
- 步骤2:之前的属性值需要重置。
- 步骤3:遍历所有的数据项,重新插入到哈希表中。
- ❓在什么时候调用扩容方法呢?
- 在每次添加完新的数据时,都进行判断。(也就是put方法中)
- 修改容量
private resize(newLimit: number) {
this.length = newLimit;
const oldStorage = this.storage;
this.storage = [];
this.count = 0;
oldStorage.forEach((bucket) => {
if (!bucket) return;
for (let i = 0; i < bucket.length; i++) {
const tuple = bucket[i];
this.put(tuple[0], tuple[1]);
}
});
}
const loadFactor = this.count / this.length;
if (loadFactor < 0.25 && this.length > 7) {
this.resize(Math.floor(this.length / 2));
}
容量质数
- ◼ 我们前面提到过,容量最好是质数。
- 虽然在链地址法中将容量设置为质数,没有在开放地址法中重要,
- 但是其实链地址法中质数作为容量也更利于数据的均匀分布。所以,我们还是完成一下这个步骤。
- ◼ 我们这里先讨论一个常见的面试题,判断一个数是质数。
- ◼ 质数的特点:
- 质数也称为素数。
- 质数表示大于1的自然数中,只能被1和自己整除的数。
- ◼ OK,了解了这个特点,应该不难写出它的算法:
- 更高效的质数判断
- 对于每个数n,其实并不需要从2判断到n-1
- 一个数若可以进行因数分解,那么分解时得到的两个数一定是一个小于等于sqrt(n),一个大于等于sqrt(n)。
- ✓ 注意: sqrt是square root的缩写,表示平方根;
- 比如16可以被分别。那么是
2*8
,2小于sqrt(16),也就是4,8大于4。而4*4
都是等于sqrt(n)
- 所以其实我们遍历到等于sqrt(n)即可
export function isPrime(num: number): boolean {
const sqrt = Math.sqrt(num);
for (let i = 2; i <= sqrt; i++) {
if (num % i === 0) return false;
}
return true;
}
扩容的质数
- 首先,将初始的limit为8,改成7
- ◼ 前面,我们有对容量进行扩展,方式是:原来的容量 x 2
- 比如之前的容量是7,那么扩容后就是14。14还是一个质数吗?
- 显然不是,所以我们还需要一个方法,来实现一个新的容量为质数的算法。
- ◼ 那么我们可以封装获取新的容量的代码(质数)
isPrime(num: number): boolean {
const sqrt = Math.sqrt(num);
for (let i = 2; i <= sqrt; i++) {
if (num % i === 0) return false;
}
return true;
}
private getNextPrime(num: number) {
let newPrime = num;
while (!this.isPrime(newPrime)) {
newPrime++;
}
return newPrime;
}
private resize(newLimit: number) {
let newPrime = this.getNextPrime(newLimit);
if (newPrime < 7) newPrime = 7;
this.length = newPrime;
const oldStorage = this.storage;
this.storage = [];
this.count = 0;
oldStorage.forEach((bucket) => {
if (!bucket) return;
for (let i = 0; i < bucket.length; i++) {
const tuple = bucket[i];
this.put(tuple[0], tuple[1]);
}
});
}
1 、【数据结构与算法——TypeScript】数组、栈、队列、链表
2 、【数据结构与算法——TypeScript】算法的复杂度分析、 数组和链表的对比