上一节中实现的SimpleHashMap,没有解决冲突的问题,这一节我们继续深入
由于table的大小是有限的,而key的集合范围是无限大的,所以寄希望于hashcode散落,肯定会出现多个key散落在同一个数组下标下面,
因此我们要引入另外一个概念,将key和value同时存入table[index]中,即将key和value构成一个对象放在table[index],而且可能存放多个,他们的key对应的index相同,但是key本身不同
现在我们就该讨论以什么样的方式存储这些散落在同一个数组下标的元素
可以考虑数组?
也可以考虑链表存储
源码里面是用链表存储的,其实我也没明白这两种方式在这里有什么区别
,感觉无论在检索和存储上都是差不多的效率,
检索都是需要遍历的方式,而存储也可以是顺序的
那这个问题留给大家吧。
我们来实现链式存储的方式,首先定义一个链表数据结构Entry:
public class Entry<K, V> {
//存储key
final K key;
//存储value
V value;
//存储指向下一个节点的指针
Entry<K, V> next;
//存储key映射的hash
final int hash;
}
新的EntryHashMap实现方式
public class EntryHashMap<K, V> {
transient Entry[] table;
transient int size;
public V put(K key, V value) {
//计算出新的hash
int hash = hash(key.hashCode());
//计算出数组小标i
int i = indexFor(hash, table.length);
//遍历table[i],如果table[i]没有与新加入的key相等的,则新加入
//一个value到table[i]中的entry,否则将新的value覆盖旧的value并返回旧的value
for (Entry<K, V> e = table[i]; e != null; e = e.next) {
Object k;
if (e.hash == hash && ((k = e.key) == key || key.equals(k))) {
V oldValue = e.value;
e.value = value;
return oldValue;
}
}
addEntry(hash, key, value, i);
return null;
}
public void addEntry(int hash, K key, V value, int bucketIndex) {
Entry<K, V> e = table[bucketIndex];
//将新的元素插入链表前端
table[bucketIndex] = new Entry<>(hash, key, value, e);
size++;
}
/**
* 通过hash code 和table的length得到对应的数组下标
*
* @param h
* @param length
* @return
*/
static int indexFor(int h, int length) {
return h & (length - 1);
}
/**
* 通过一定算法计算出新的hash值
*
* @param h
* @return
*/
static int hash(int h) {
h ^= (h >>> 20) ^ (h >>> 12);
return h ^ (h >>> 7) ^ (h >>> 4);
}
}
分享到:
相关推荐
Java源代码是编程世界的基石,它是Java程序员用Java语言编写的程序文本,包含了类、方法、变量等元素,是理解程序逻辑和功能的核心。在Java编程中,源代码通常以`.java`为扩展名,经过Java编译器的处理,会被转化为...
java代码-使用java解决手写hashMap的源代码 ——学习参考资料:仅用于个人学习使用!
4. **集合框架**:Java集合框架包括ArrayList、LinkedList、HashSet、HashMap等,它们在源代码中会被广泛应用,用于数据的存储和操作。 5. **输入/输出流**:Java的I/O流系统广泛用于文件读写、网络通信等场景,源...
"java简单实例程序源代码"这个压缩包包含了一系列章节相关的Java实例源代码,适合初学者和有经验的开发者用来加深对Java语言的理解。以下是这些章节可能涉及的重要知识点的详细解释: 1. **CH11**: 这个章节可能...
这个压缩包包含了140个经典的Java源代码程序,涵盖了各种基础到进阶的编程概念。下面,我们将详细讨论这些知识点。 1. **基础语法**:作为初学者,首先需要掌握Java的基础语法,包括变量声明、数据类型(如int、...
"250个Java源代码"这个压缩包很显然是为了帮助初学者和希望提升Java技能的开发者提供实践素材。这些源代码实例涵盖了Java的基础概念到进阶特性,是学习和理解Java语法、编程技巧以及解决问题的有效工具。 首先,...
Java源代码包含了JDK(Java Development Kit)的所有组件,包括核心类库如java.lang、java.util、java.io等,这些类库构成了Java平台的基础。通过阅读源代码,我们可以了解到Java的API是如何实现的,比如对象的创建...
通过阅读源代码,开发者可以深入了解内部的工作原理,提升编程技能,并能够更好地利用Java 8的功能来解决实际问题。与Eclipse等IDE配合使用,可以方便地查看和调试源代码,进一步提高开发效率。
在IT领域,尤其是Java编程语言的学习与开发过程中,掌握经典源代码是提升技能的重要途径。"JAVA 经典源代码"这个主题包含了丰富的知识内容,它涵盖了基础语法、设计模式、数据结构、算法等多个方面。这里我们将深入...
在描述中提到的".java"文件是Java源代码文件,它是编写Java程序的基本单元。每个.java文件通常包含一个或多个类定义,编译后会生成对应的字节码文件(.class),由Java虚拟机执行。".java"文件的实用性质意味着它们...
Java源代码涉及到的基础知识包括变量、数据类型、运算符、流程控制(如if语句、for循环、while循环)、方法定义和调用。在applet相关的源代码中,可以学习到如何在Web环境中运行Java小程序,包括Applet类、初始化和...
Java API源代码是Java开发中的核心组成部分,它包含了Java标准库中的所有类和接口,这些类和接口构成了Java平台的基础。Sun Microsystems(后被Oracle收购)是Java的原始开发者,他们发布的源代码对于深入理解Java的...
Java核心源代码是Java编程语言的核心组成部分,它包括了大量的类库和API,这些构成了Java开发的基础框架。在Java中,"以java开头的包"通常指的是Java标准库,这是一个庞大的集合,涵盖了各种功能,从基本的数据类型...
java源代码原理,深入理解HashMap原理、高可用redis集群架构、zookeeper分布式、海量数据缓存、springAOP底层源码分析等
3. **异常处理**:Java的异常处理机制是其强大之处,源代码中可能会包含try-catch-finally块,学习者可以从中学习如何处理运行时可能出现的错误。 4. **集合框架**:Java集合框架包括ArrayList, LinkedList, ...
HashMap 的存储实现可以通过查看其 put(K key, V value) 方法的源代码来了解。该方法首先判断 key 是否为 null,然后根据 key 的 hashCode 值计算 Hash 值,接着搜索指定 hash 值在对应 table 中的索引,如果 i 索引...
Java2源代码是Java编程语言的一个重要里程碑,它包含了Java平台标准版(Java SE)的第二代核心特性。这个压缩包文件很可能包含了丰富的示例代码、教程和项目的源码,帮助开发者深入理解Java语言的关键概念和技术。...
《Head First Java源代码 (中文第2版)》是一本为初学者设计的Java编程教材,书中通过生动、直观的方式讲解了Java编程的基础知识。源代码的提供是为了方便读者实践和理解书中所讲述的编程概念。以下是根据标题、描述...
《Java案例开发集锦(第二版)》是Java学习者的一份宝贵资源,它由袁然、郑自国等多位专业作者共同编写,旨在通过实际的项目案例来深入讲解Java编程技术。这份源代码集合包含了书中所有示例的实现,为初学者提供了直观...
通过阅读和实践这些源代码,你可以深入学习Java的各种特性,包括但不限于: 1. 类和对象:Java是面向对象的语言,源代码会展示如何定义类,创建对象,以及如何利用继承、封装和多态等面向对象原则来设计软件。 2. ...