哈希表 谁践踏了优雅 2023-01-13 04:14 48阅读 0赞 ### 哈希表 ### * 概念 * 哈希函数设计 * 冲突 * * 调节负载因子 * 冲突解决 * * 闭散列 * 开散列/哈希桶 * 实现自己的哈希表: # 概念 # 理想的搜索方法:不经过任何比较,一次直接从表中得到想要的搜索元素。 构造某种存储结构,通过某种函数(hashfunc)使元素的存储位置与他的关键key之间建立一一映射的关系,在查找的时候,可以很快找到该元素。 哈希方法中使用的转换函数称为哈希(散列)函数,构造出的数据结构称为哈希表(HashTable)(后者称为散列表) # 哈希函数设计 # 常见的哈希函数: 1、直接定制法: Hash(key)= A\*key+B 优点:简单均匀 2、除留余数法——(常用) Hash(key) = key%p(p<=m) 3、 … # 冲突 # 冲突的发生是必然的,我们要做的是降低冲突率 例如使用除留余数法,会出现相同的余数的情况,(出现相同的哈希地址),这种情况称为哈希冲突(哈希碰撞) 避免冲突的方法: 1、设计好的哈希函数 2、负载因子的调节 ## 调节负载因子 ## 负载因子的计算是通过: 填入表中的元素 / 散列表的长度![在这里插入图片描述][watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl81MjE0MjczMQ_size_16_color_FFFFFF_t_70] 已知哈希表中的关键字个数是不可变的,那么我们只能调整哈希表中数组的大小。 # 冲突解决 # ## 闭散列 ## 闭散列:也叫开放地址法,当发生冲突的时候,如果哈希表未被填满,那么就说明哈希表中还有空位,就将key放在冲突位置的“下一个”空位中。 ## 开散列/哈希桶 ## 开散列法,又叫链地址法,首先通过关键码集合用散列函数计算地址,将相同地址关键码放在同一子集合中,每个子集合称一个桶,各个桶中的元素通过一个单链表链接起来,各链表的头结点存在哈希表中。 简单理解就是: 一个数组,数组中存存链表的头结点,发生冲突的元素放在链表中。 注意: 假如元素发生聚集,那么就会退回成链表,所以需要我们合理解决冲突。 通常情况下我们认为哈希表的冲突率不高,冲突个数是可控的,每个桶中的链表长度是一个常数,所以通常意义上,我们认为哈希表的插入/删除/查找的时间复杂度是O(1) # 实现自己的哈希表: # 结点: package HashTable; public class Node { int key; Node next; public Node(int key) { this.key = key; } @Override public String toString() { return "Node{" + "val=" + key + ", next=" + next + '}'; } } MyHashTable: package HashTable; public class MyHashTable { //存头结点的数组 Node[] array; int size; public MyHashTable() { array = new Node[11]; size = 0; } //冲突因子 private double LOAD_FACTOR = 0.75; //插入——头插 public boolean insert(int key) { if (find(key)) { return false; } int index = key % array.length; Node node = new Node(key); node.next = array[index]; array[index] = node; size++; if ((double) size / array.length > LOAD_FACTOR) { grow(); } return true; } //扩容 //2倍扩容 private void grow() { Node[] newArray = new Node[array.length * 2]; for (Node head : array) { Node cur = head; while (cur != null) { int index = cur.key % newArray.length; Node next = cur.next; cur.next = newArray[index]; newArray[index] = cur; cur = next; } } array = newArray; } //删除 public boolean remove(int key) { int index = key % array.length; Node cur = array[index]; Node prev = null; while (cur != null) { if (key == cur.key) { if (prev != null) { prev.next = cur.next; } else { array[index] = array[index].next; } size--; return true; } prev = cur; cur = cur.next; } return false; } //查找 public boolean find(int key) { int index = key % array.length; for (Node cur = array[index]; cur != null; cur = cur.next) { if (cur.key == key) { return true; } } return false; } } Test: package HashTable; import java.util.Random; public class Test { public static void main(String[] args) { Random random = new Random(2020412); MyHashTable hashTable = new MyHashTable(); for (int i = 0; i < 10000; i++) { int key = random.nextInt(); if (key >= 0 && !hashTable.find(key)) { hashTable.insert(key); } } int min = Integer.MAX_VALUE; int max = Integer.MIN_VALUE; int sum = 0; for (int i = 0; i < hashTable.array.length; i++) { Node cur = hashTable.array[i]; int count = 0; while (cur != null) { count++; cur = cur.next; } if (count < min) { min = count; } if (count > max) { max = count; } sum += count; } System.out.println(hashTable.size); System.out.println(hashTable.array.length); System.out.println(max); System.out.println(min); System.out.println((double) sum / hashTable.array.length); } } ![在这里插入图片描述][watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl81MjE0MjczMQ_size_16_color_FFFFFF_t_70 1] [watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl81MjE0MjczMQ_size_16_color_FFFFFF_t_70]: /images/20221022/49f2ef6d9aab4e4aaa5b64afb619f22d.png [watermark_type_ZmFuZ3poZW5naGVpdGk_shadow_10_text_aHR0cHM6Ly9ibG9nLmNzZG4ubmV0L3dlaXhpbl81MjE0MjczMQ_size_16_color_FFFFFF_t_70 1]: /images/20221022/3e6dcaf1679c4ca790f2bf793b8611d4.png
相关 哈希值 哈希表_哈希杰森 哈希值 哈希表 我最近写了一个[简单的库,可预测地对json进行哈希处理][json] 。 该实用程序基于出色的[Jackson Json解析库][Jackson Json 阳光穿透心脏的1/2处/ 2023年02月25日 04:56/ 0 赞/ 52 阅读
相关 哈希表 ![Center][] [Center]: /images/20220731/1379dbdc6efb4a42a0b011f0b3aa4455.png 「爱情、让人受尽委屈。」/ 2022年08月14日 04:56/ 0 赞/ 229 阅读
相关 哈希表 什么是哈希表 哈希表(Hash table,也叫散列表),是根据关键码值(Key value)而直接进行访问的[数据结构][Link 1]。也就是说,它通过把关键码 悠悠/ 2022年07月15日 12:14/ 0 赞/ 247 阅读
相关 哈希表 散列表(Hash table,也叫哈希表),是根据关键码值(Key value)而直接进行访问的数据结构。也就是说,它通过把关键码值映射到表中一个位置来访问记录,以加快查找的速 系统管理员/ 2022年06月10日 01:26/ 0 赞/ 304 阅读
相关 哈希表 我们知道,通过对数组进行直接寻址(Direct Addressing),可以在 O(1) 时间内访问数组中的任意元素。所以,如果存储空间允许,可以提供一个数组,为每个可能的关键 快来打我*/ 2022年06月05日 02:20/ 0 赞/ 415 阅读
相关 哈希表 哈希表是种数据结构,它可以提供快速的插入操作和查找操作。第一次接触哈希表时,它的优点多得让人难以置信。不论哈希表中有多少数据,插入和删除(有时包括侧除)只需要接近常量的时间即0 今天药忘吃喽~/ 2022年02月01日 14:36/ 0 赞/ 459 阅读
相关 【哈希表】 char FirstNotRepeatingChar(char pString) { // invalid input if(! r囧r小猫/ 2022年01月06日 11:33/ 0 赞/ 378 阅读
相关 哈希表 一、简介 如果所有的键都是小整数,那么我们可以用一个数组来实现无序的符号表,将键作为数组的索引i而数组中i(键)处储存的就是对应的值。 这样就可以快速地访问任意键的值, 旧城等待,/ 2021年12月22日 01:21/ 0 赞/ 455 阅读
相关 哈希表 【一】哈希表 > 他通过把关键码值映射到表中的一个位置来访问记录,以加快查找的速度。这个映射函数就是散列函数。 ![watermark_type_ZmFuZ3poZW5na 傷城~/ 2021年08月11日 17:13/ 0 赞/ 610 阅读
还没有评论,来说两句吧...