【数据结构与算法】HashTable相关操作实现(附完整源码)

转载请注明出处:http://blog.csdn.net/ns_code/article/details/20763801


前言

    学过Java的人肯定对Hash这个词非常之熟悉,HashTable、HashSet、HashMap等都是对哈希表的封装或改进。这次我们来看下哈希表用C语言表示的封装实现。


哈希表

    哈希表又叫散列表,是实现字典操作的一种有效数据结构。哈希表的查询效率极高,在没有冲突(后面会介绍)的情况下不需经过任何比较,一次存取便能得到所查记录,因此理想情况下,查找一个元素的平均时间为O(1)。

    哈希表就是描述key—value对的映射问题的数据结构,这在Java中大家都知道,更详细的描述是:在记录的存储位置和它的关键字之间建立一个确定的对应关系f,使每个关键字与哈希表中唯一一个存储位置相对应。我们称这个对应关系f为哈希函数,这个存储结构即为哈希表。

    直接寻址表

    当关键字的全域U比较小时,直接寻址是一种简单而有效的技术,它的哈希函数很简单:f(key) = key,即关键字大小直接与元素所在的位置序号相等。另外,如果关键字不是自然数,我们需要通过某种手段将其转换为自然数,比如可以将字符关键字转化为其在字母表中的序号作为关键字。直接寻址法不会出现两个关键字对应到同一个地址的情况,既不会出现f(key1) = f(key2)的情况,因此不用处理冲突,这便是其优点所在。

    散列表

    直接寻址的缺点非常明显,如果全域U很大,则在一台标准的计算机可用内存容量中,要存储大小为U的一张表也许不太实际,而且,实际需要存储的关键字集合K可能相对U来说很小,这时散列表需要的存储空间要比直接表少很多。散列表通过散列函数f计算出关键字key在槽的位置。散列函数f将关键字域U映射到散列表T[0...m-1]的槽位上。但是这里会存在一个问题:若干个关键字可能映射到了表的同一个位置处(算法导论上名其曰“”),我们称这种情形为冲突。当然理想的方法是尽量避免冲突,我们可以尽可能第将关键字通过f随即地映射到散列表的每个位置上。


哈希函数

  • 13
    点赞
  • 47
    收藏
    觉得还不错? 一键收藏
  • 20
    评论
实验目的: 掌握使用哈希表解决只出现一次的数字问题的基本思路与方法。 实验原理: 只出现一次的数字问题可以使用哈希表来解决。具体步骤如下: 1. 遍历整个数组,将每个数字作为 key,出现的次数作为 value 存入哈希表中。 2. 再次遍历整个数组,找到 value 为 1 的 key,即为只出现一次的数字。 3. 如果哈希表中没有 value 为 1 的 key,说明没有只出现一次的数字。 实验步骤: 1. 创建一个哈希表,将数组中每个数字作为 key,出现的次数作为 value 存入哈希表中。 2. 遍历哈希表,找到 value 为 1 的 key,即为只出现一次的数字。 3. 如果哈希表中没有 value 为 1 的 key,说明没有只出现一次的数字。 实验代码实现: ```python def find_only_once(nums): hashtable = {} # 将每个数字作为 key,出现的次数作为 value 存入哈希表中 for num in nums: if num not in hashtable: hashtable[num] = 1 else: hashtable[num] += 1 # 遍历哈希表,找到 value 为 1 的 key,即为只出现一次的数字 for key, value in hashtable.items(): if value == 1: return key # 如果哈希表中没有 value 为 1 的 key,说明没有只出现一次的数字 return None ``` 实验结果: 对于数组 [1, 2, 2, 3, 3, 4, 4, 5, 5],只出现一次的数字为 1。 对于数组 [1, 1, 2, 2, 3, 3, 4, 4, 5],只出现一次的数字为 5。 对于数组 [1, 1, 2, 2, 3, 3, 4, 4, 5, 5],没有只出现一次的数字。 实验结论: 使用哈希表可以解决只出现一次的数字问题,时间复杂度为 O(n),空间复杂度为 O(n)。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 20
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值